我试图降低所有扩展的大小写,不管它是什么.到目前为止,根据我所看到的,您必须指定要转换为小写的文件扩展名.然而,我只想把名字中第一个最后一个点.后面的所有内容都小写.

我怎么能在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个进程是没有效率的.这是正确的,以前的循环版本会更好.

Linux相关问答推荐

如何检测文件系统是否支持权限?

Git - 打印以不同编码混合的文件

使用awk命令将以:分隔的两个文件合并的方法

需要获取文件名、行号、模式在文件中的匹配位置

如何使用 shell 脚本将文本文件转换为 JSON 文件

Bash 更新 yaml 文件中的图像值

如何让 Flutter 用鼠标拖动而不是滚轮滚动? (Linux)

`std::cout` 是如何实现的?

有没有办法定义自定义隐式 GNU Make 规则?

Dockerfile:无法复制文件

tmux:挂起不加载,不响应任何选项命令

如何使用不同的出口 IP 一次运行多个 Tor 进程?

如何更改某些文件模式/扩展名的权限?

如何在没有 IDE 的情况下构建和部署三星 SmartTV 应用程序(例如:在 Linux 上)

使用 ssh 判断远程主机上是否存在文件

何时使用管道与何时使用共享内存

如何显示正在运行的进程列表 Python?

Android - 找不到命令

如何在 IE 中使用 Linux 进行测试

Linux 的文本编辑器(除了 Vi)?