解
您可以在一行中解决任务:
find . -name '*.*' -exec sh -c '
a=$(echo "$0" | sed -r "s/([^.]*)\$/\L\1/");
[ "$a" != "$0" ] && mv "$0" "$a" ' {} \;
注意:对于包含换行符的文件名,这将中断。但现在忍受我。
使用示例
$ 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
说明
您在当前目录(。)中找到具有句点的所有文件。在其名称(-name’*。*’),并运行每个文件的命令:
a=$(echo "$0" | sed -r "s/([^.]*)\$/\L\1/");
[ "$a" != "$0" ] && mv "{}" "$a"
该命令意味着:尝试将文件扩展名转换为小写(使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。
它使脚本安全,因为在这种情况下,shell {{}作为数据,而不是作为代码部分,直接在命令行中指定。
(我感谢@gniourf_gniourf指向我这个真正重要的问题)。
如你所见,如果在脚本中直接使用{},
它可能有
一些shell注入在文件名中,像:
; 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(像GNU find)和Bash(所以read支持-d分隔符切换)的find:
find . -name '*.*' -print0 | while IFS= read -r -d '' f
do
a=$(echo "$f" | sed -r "s/([^.]*)\$/\L\1/");
[ "$a" != "$f" ] && mv "$f" "$a"
done
如果你真的想要一个foolproof方法(和你应该!),使用最近版本的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
回到原来的解决方案
或者在一个发现(回到原来的解决方案,一些修复,使它真的万无一失):
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进程效率不高。这是正确的,以前的循环版本会更好。