我想编写一个脚本,递归地遍历输入目录,并为每个文件在单独的目录层次结构下生成相应的文件。
例如,文件input/a/b/c.txt
变为output/a/b/c.txt
并input/a/d.txt
变为output/a/d.txt
.当我执行此操作时,用输出目录的名称替换输入目录的名称的干净方法是什么?
我所知道的两种方式感觉都不对。第一种是使用#
参数扩展来去掉文件前缀。但是,如果输入目录的名称包含特殊字符(例如*
.如果输入目录的名称位于变量中,也很难读取。
find input -type f | while read -r infile; do
outfile=output/"${infile#input/}"
mkdir -p "$(dirname "$outfile")"
some-program "$infile" > "$outfile"
done
我尝试的另一件事是 cd 到输入目录并对所有内容使用绝对链接。但是,这需要多次调用cd
并且还取决于readlink
,这不是最可移植的。
abspwd=$(pwd)
absinput=$(readlink -f input)
absoutput=$(readlink -f output)
cd "$absinput"
find . -type f | while read infile; do
outfile="$absoutput/$infile"
mkdir -p "$(dirname "$outfile")"
some-program "$infile" > "$outfile"
done
cd "$abspwd"
还有别的办法吗?感觉应该有一种更简单的方法来做到这一点。
答案1
你就快到了。
我将放置一个 rsync,它将仅复制输出中输入的目录结构。然后,当您的 find-while 管道运行时,它将把某个程序的输出扔到正确的位置。
TOP="input"
rsync -a -f'+ */' -f'- *' "$TOP"/ output/
find "$TOP" -type f | while IFS= read -r infile; do
outfile=output/${infile#"$TOP"/}
some-program "$infile" > "$outfile"
done
注意带有换行符的文件名将不会被正确处理。如果您认为这种情况是有必要的,那么可以使用更多代码来处理它们。我不想分散您的注意力,即将文件名从一个目录迁移到另一个目录。