Bash 通过解析 find 的输出来移动和重命名

Can*_*low 3 linux bash command-line find mv

我嵌套了 pdf 文件的目录,我想将它们提取到更高级别的目录中,将它们重命名如下:

我的文件是这样的:

./path1/pathA/fileI.pdf
./path1/pathB/fileII.pdf
Run Code Online (Sandbox Code Playgroud)

我想实现:

./path1_pathA_fileI.pdf    
./path1_pathB_fileII.pdf
Run Code Online (Sandbox Code Playgroud)

我知道我可以做一个文件列表

find . -type f -name "*.pdf"

我可以想象一个解决方案使用

find . -type f -name "*.pdf" | mv -t ...

但是我不知道如何填写...因为我不了解bash中的解析和变量赋值。如何在“/”处拆分路径并形成如上所述的新路径和文件名?

提前谢谢了!

Joh*_*024 7

尝试:

find . -mindepth 2 -type f -name '*.pdf' -exec bash -c 'f=${1#./}; mv "$1" "./${f//\//_}"' None {} \;
Run Code Online (Sandbox Code Playgroud)

这对所有文件名都是安全的,即使是名称中有换行符的文件名。

这个怎么运作

  • -mindepth 2

    这告诉 find 不要处理当前目录中已有的任何文件。

  • -type f -name '*.pdf'

    这将搜索限制为带有pdf扩展名的常规文件。

  • -exec bash -c '...' None {} \;

    这将在提供文件名作为第一个参数的引用字符串中运行命令$1

    对于我们的目的,字符串None只是一个占位符。它被分配给$0,根据 bash 约定,是我们正在运行的命令的名称。

  • f=${1#./}; mv "$1" "./${f//\//_}"

    这 (a)./从文件名中删除前缀,以及 (b) 使用新名称将文件移动到所需位置。

    ${1#./}是 bash前缀删除的一个例子。它返回strign$1./从一开始就删除。${f//\//_}是 bash模式替换的一个例子。它返回$f所有/替换为的字符串_。要了解有关这些功能的更多信息,请参阅man bash标题为Parameter Expansion的部分。

更高效的版本

上述版本为找到的每个文件调用 bash。或者,我们可以为找到的几个文件只调用一次 bash。为此,我们将命令包装在一个for循环中:

find . -mindepth 2 -type f -name '*.pdf' -exec bash -c 'for f in "$@"; do f=${f#./}; echo mv "$f" "./${f//\//_}"; done' None {} +
Run Code Online (Sandbox Code Playgroud)

替代问题

假设我们想要的所有文件都在第二级目录中,并且我们希望移动的文件的目录名称顺序颠倒,以便./path1_pathA_fileI.pdf我们最终得到./pathA_path1_fileI.pdf. 在这种情况下:

for d1 in */; do d1=${d1%/}; for d2 in "$d1"/*/; do d2=${d2%/}; p="${d2#$d1/}_$d1"; for f in "./$d2"/*.pdf; do echo mv "$f" "./${p}_${f#./$d2/}"; done; done; done
Run Code Online (Sandbox Code Playgroud)

或者,对于那些喜欢将命令分散到多行的人:

for d1 in */
do
    d1=${d1%/}
    for d2 in "$d1"/*/
    do
         d2=${d2%/}
         p="${d2#$d1/}_$d1"
         for f in "./$d2"/*.pdf
         do
             echo mv "$f" "./${p}_${f#./$d2/}"
         done
    done
done
Run Code Online (Sandbox Code Playgroud)