通常,我们可以将命令输出重定向到文件,如下所示:

cat a.txt >> output.txt

如我所试,如果cat失败,仍然会创建output.txt,这不是我期望的。我知道我可以这样测试:
if [ "$?" -ne "0"]; then
    rm output.txt
fi

但是,当我执行output.txt之前已经有这样的cat时,这可能会导致一些开销问题。
所以我还需要在output.txt之前存储cat状态,如果在执行cat之前已经有这样的output.txt,我不应该错误地删除output.txt……但是在竞争条件下仍然可能存在问题,如果有其他进程创建此输出怎么办? .txt就在我的cat前面吗?

那么,有什么简单的方法可以使如果命令失败,重定向output.txt将被删除,甚至不被创建?

最佳答案

固定的输出文件名是个坏消息。不要使用它们。

您可能应该重新设计处理过程,以便获得带有日期戳的文件名。如果失败,则应使用mktemp命令创建一个临时文件,将要执行的命令写入该文件,当命令成功执行后,您可以将该临时文件移至“最终”输出-并且可以自动清理暂时的失败。

outfile="./output-$(date +%Y-%m-%d.%H:%M:%S).txt"
tmpfile="$(mktemp ./gadget-maker.XXXXXXXX)"

trap "rm -f '$tmpfile'; exit 1" 0 1 2 3 13 15

if cat a.txt > "$tmpfile"
then mv "$tmpfile" "$outfile"
else rm "$tmpfile"
fi

trap 0

如果您坚持,可以将outfile简化为output.txt(但这并不安全)。您可以在mktemp命令中使用所需的任何前缀。请注意,通过在当前目录中创建临时文件(也将在该目录中创建最终输出文件),您可以避免在操作的mv阶段复制跨设备文件-它是link()unlink()系统调用(甚至可能是rename()系统调用(如果您的计算机上存在此类事物;仅在Mac OS X上存在)。

10-07 16:24
查看更多