当递归遍历输入目录时,如何用输出目录替换输入目录?

当递归遍历输入目录时,如何用输出目录替换输入目录?

我想编写一个脚本,递归地遍历输入目录,并为每个文件在单独的目录层次结构下生​​成相应的文件。

例如,文件input/a/b/c.txt变为output/a/b/c.txtinput/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

注意带有换行符的文件名将不会被正确处理。如果您认为这种情况是有必要的,那么可以使用更多代码来处理它们。我不想分散您的注意力,即将文件名从一个目录迁移到另一个目录。

相关内容