在许多 Linux 书籍中都写到“重定向在命令之前执行”。让我们考虑一下:
$ cat bar
hello
$ ecno "foo" > bar
bash: ecno: command not found
$cat bar
- 如果 bar 不存在,重定向将创建它。
- 如果 bar 存在并且命令不起作用,它会删除 bar 的内容。
但是它并没有将任何输出复制到bar中(因为没有输出),所以应该说'重定向在命令之前部分执行',因为'>'的一部分在这里不起作用,即复制输出将命令放入文件栏中。这样对吗?
从 bash 手册第 3.6.2 节:
重定向总是在执行命令之前
> file
截断文件,这意味着无论输出如何,内容都会被删除;如果有实际输出(最明显的是没有调用 in )并且没有发生进一步的错误, 则实际写入文件将发生。write()
strace -f -e open,dup2,write bash -c 'true > out2.txt'
因此,一切正常,按照规范,措辞是正确的。事实上,这种行为是由POSIX Shell 命令语言规范指定的,并且受到所有符合 POSIX 的 shell 的支持,包括
ksh
和dash
(也称为 Ubuntu 的/bin/sh
,请参阅相关问题)。在系统层面,重定向是由系统调用的 dup 家族执行的,这就是为什么重定向被正式称为复制文件描述符,并且在我们执行
>/dev/null 2&>1
重定向类型时最为突出。strace
在以下成功和失败命令的示例中,可以通过命令观察到。请注意如何使用O_CREAT|O_TRUNC
标志打开文件。好吧,第二部分起作用了,因为它将空输出复制到文件中。
例如,如果您使用
2>
而不是 stdout 重定向 stderr,您会看到错误使其进入该文件:并且“bar”的内容更新为:
文件被截断的事实也是重定向工作方式的一部分。
我希望这可以帮助您了解重定向的工作原理!