linux - bash重定向到同一个文件的一致性

标签 linux bash shell

我想将某些命令的标准输出和标准错误重定向到作为参数传递给我的脚本的文件。参数是可选的,默认情况下它们是 /dev/stdout/dev/stderr

这就是为什么我用以下方式重定向 some-command 1>$myStdout 2>$myStderr。因为我事先不知道 myStdoutmyStderr 是否相同,所以我不能使用这种 2>&1 的重定向形式.

问题来了。如果 myStdoutmyStderr 指向同一个文件,我可以丢失一些输出。考虑以下脚本

>&2 echo "err" &
echo "out"

如果我像 test.sh 1>log 2>&1 这样运行它,那么我会在日志文件中看到这两行,但是如果我像 test.sh 1>log 2> 这样运行它log 然后我只看到 err

这是第一个问题:为什么会这样? bash 如何处理上述示例中的并发写入?

我该如何解决?我知道我可以检查文件是否相同

if [[ $myStdout and $myStderr are the same ]]; then
  redirect using 1>$myStdout 2>&1
else
  redirect using 1>$myStdout 2>$myStderr
fi

但是这种方式很丑陋。我必须以某种方式检查文件是否相同(并处理符号链接(symbolic link)解析)并且我必须复制相当大的命令。

最佳答案

如果单独打开描述符,它们是完全独立的流。如果它们指向同一个文件,它们将各自独立地跟踪文件偏移量。所以第一次写入一个流时它会写入文件的开头,而当你第一次写入另一个流时它也会写入文件的开头,覆盖之前写入的内容。这种情况会继续发生,对每个流的写入会覆盖另一个流。

你可以做的是首先清空每个文件,然后以追加模式打开它们:

> $myStdout
> $myStderr
redirect 1>> $myStdout 2>> $myStderr

通过追加,每次写入都将首先查找文件末尾,因此它会追寻上一次写入的内容,而不是覆盖它。

但是,可能还有另一个问题。如果程序使用 stdio,它默认缓冲 stdout,但不缓冲 stderr。因此,除非程序在打印错误消息之前刷新 stdout 缓冲区,否则它们很可能散布在输出中的随机位置,而不是整齐地排在自己的行上。

关于linux - bash重定向到同一个文件的一致性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29295842/

相关文章:

php - 如何从符号链接(symbolic link)获取文件路径?

java - SWT DateTime 不考虑语言环境

linux - 使 bash 脚本在 Linux 和 FreeBSD 之间可移植的正确方法是什么?

bash - 在 bash 4 中递增变量会触发 EXIT,但在 bash 3 中不会

bash - shell : How to print progressbar

java - OSX 如何让 Shell 脚本将目录更改为脚本所在的目录?

bash - sed 替换每行内的新行字符

android - 如何使用 WiFi direct 将一部 Android 手机连接到多部其他 Android 手机,并且每部手机都使用单独的 channel ?

linux - 在 rm -rf 语句中使用环境变量是不好的做法吗?

c - ifconfig 和套接字 inet_ntop 的不同 IP 结果