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中的解析和变量赋值。如何在“/”处拆分路径并形成如上所述的新路径和文件名?
提前谢谢了!
尝试:
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)
归档时间: |
|
查看次数: |
1350 次 |
最近记录: |