通常,我们可以将命令输出重定向到文件,如下所示:
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上存在)。