我试图降低所有扩展的大小写,不管它是什么.到目前为止,根据我所看到的,您必须指定要转换为小写的文件扩展名.然而,我只想把名字中第一个最后一个点.
后面的所有内容都小写.
我怎么能在bash
分钟内做到这一点?
我试图降低所有扩展的大小写,不管它是什么.到目前为止,根据我所看到的,您必须指定要转换为小写的文件扩展名.然而,我只想把名字中第一个最后一个点.
后面的所有内容都小写.
我怎么能在bash
分钟内做到这一点?
Solution
您可以在一行中解决该任务:
find . -name '*.*' -exec sh -c '
a=$(echo "$0" | sed -r "s/([^.]*)\$/\L\1/");
[ "$a" != "$0" ] && mv "$0" "$a" ' {} \;
注意:对于包含换行符的文件名,这将中断.但现在请容忍我.
Example of usage
$ mkdir C; touch 1.TXT a.TXT B.TXT C/D.TXT
$ find .
.
./C
./C/D.TXT
./1.TXT
./a.TXT
./B.TXT
$ find . -name '*.*' -exec sh -c 'a=$(echo "$0" | sed -r "s/([^.]*)\$/\L\1/"); [ "$a" != "$0" ] && mv "$0" "$a" ' {} \;
$ find .
.
./C
./C/D.txt
./a.txt
./B.txt
./1.txt
Explanation
找到当前目录(.
)中名称(-name '*.*'
)中包含句点.
的所有文件,并对每个文件运行命令:
a=$(echo "$0" | sed -r "s/([^.]*)\$/\L\1/");
[ "$a" != "$0" ] && mv "{}" "$a"
该命令的意思是:try 将文件扩展名转换为小写(即sed
):
$ echo 1.txt | sed -r "s/([^.]*)\$/\L\1/"
1.txt
$ echo 2.TXT | sed -r "s/([^.]*)\$/\L\1/"
2.txt
并将结果保存到a
变量中.
如果某些内容更改为[ "$a" != "$0" ]
,请重命名文件mv "$0" "$a"
.
正在处理的文件名({}
)作为其附加参数传递给sh -c
,在命令行中显示为$0
.
如你所见,如果你在脚本中直接使用{}
,
; rm -rf * ;
在这种情况下,壳牌将注入视为
While-version
更清晰,但稍长一点的脚本版本:
find . -name '*.*' | while IFS= read -r f
do
a=$(echo "$f" | sed -r "s/([^.]*)\$/\L\1/");
[ "$a" != "$f" ] && mv "$f" "$a"
done
对于包含换行符的文件名,这仍然会中断.要解决此问题,您需要一个支持-print0
(如GNUfind
)和Bash的find
(因此read
支持-d
分隔符switch ):
find . -name '*.*' -print0 | while IFS= read -r -d '' f
do
a=$(echo "$f" | sed -r "s/([^.]*)\$/\L\1/");
[ "$a" != "$f" ] && mv "$f" "$a"
done
对于包含尾随换行符的文件(因为它们将被a=$(...)
个子shell吸收).如果你想要一个万无一失的方法(你应该!),这仍然会中断,使用最新版本的Bash(Bash≥4.0)支持,,
参数扩展,以下是最终解决方案:
find . -name '*.*' -print0 | while IFS= read -r -d '' f
do
base=${f%.*}
ext=${f##*.}
a=$base.${ext,,}
[ "$a" != "$f" ] && mv -- "$f" "$a"
done
Back to the original solution
或者在one find
go中(返回原始解决方案,并进行一些修复,使其真正万无一失):
find . -name '*.*' -type f -exec bash -c 'base=${0%.*} ext=${0##*.} a=$base.${ext,,}; [ "$a" != "$0" ] && mv -- "$0" "$a"' {} \;
我添加了-type f
,以便只重命名常规文件.如果不这样做,如果在文件名之前重命名目录名,仍然可能会出现问题.如果还想重命名目录(以及链接、管道等),则应使用-depth
:
find . -depth -name '*.*' -type f -exec bash -c 'base=${0%.*} ext=${0##*.} a=$base.${ext,,}; [ "$a" != "$0" ] && mv -- "$0" "$a"' {} \;
所以find
执行深度优先搜索.
你可能会说,为找到的每个文件生成bash
个进程是没有效率的.这是正确的,以前的循环版本会更好.