一、关于常用文件描述符
在 shell 程序中,最常使用的 fd (file descriptor) ,文件描述符大概有三个, 分别是:
0 是一个文件描述符,表示标准输入(stdin)
1 是一个文件描述符,表示标准输出(stdout)
2 是一个文件描述符,表示标准错误(stderr)
在标准情况下, 这些fd分别跟如下设备关联:
stdin(0): keyboard 键盘输入,并返回在前端
stdout(1): monitor 正确返回值 输出到前端
stderr(2): monitor 错误返回值 输出到前端
二、举例说明
假设我们当前目录只有一个文件 test01.txt
# ll test01.txt
# ll test02.txt
【ls: 无法访问test02.txt: 没有那个文件或目录】,是因为没有test02.txt,返回错误,就是所谓的错误输出stderr(2)。
【-rw-r--r--. 1 root root 0 3月 30 12:57 test01.txt】,就是所谓的标准输出stdout(1)。
# ll test01.txt test02.txt 1>file.out 2>file.err
命令执行,没有任何返回值,原因是,返回值都重定向到相应的文件中了,而不再在前端显示。
# cat file.out
# cat file.err
一般来说, "1>" 通常可以省略成 ">",那么上述命令可以写成:
# ll test01.txt test02.txt >file.out 2>file.err
基于上述认识,我们才能理解 "1>&2" 和 "2>&1".。
1>&2 正确返回值传递给2输出通道, &2表示2输出通道 ,如果此处错写成 1>2, 就表示把1输出重定向到文件2中了。
2>&1 错误返回值传递给1输出通道,&1表示1输出通道。
# ll test01.txt test02.txt 1>file.out 2>&1
# cat file.out
现在,正确的输出和错误的输出都定向到了file.out这个文件中, 而不显示在前端。
当然, 输出不只1和2, 还有其他的类型, 这两种只是最常用和最基本的类型。
三、关于重定向
# ll test01.txt test02.txt > file.txt
> 是重定向符,就是把前面输出的内容重定向到后面指定的位置。
> 符号前前可以加数字来说明把什么内容重定向到文件中,默认是把标准输出重定向到文件,错误信息不输出到文件。
& 是一个描述符,如果1或2前不加&,会被当成一个普通文件。
2>&1 意思是把标准错误输出重定向到标准输出,先定义了标准输出到某处。
1>&2 意思是把标准输出重定向到标准错误,先定义了标准错误输出到某处。
&> filename 意思是把标准输出和标准错误输出都重定向到文件filename中,就无所谓先定义哪个输出了。
# ll test01.txt test02.txt >file.out 2>&1
# ll test01.txt test02.txt 2>file.err 1>&2
# ll test01.txt test02.txt &>file.txt
那么重定向中 >&2 怎么理解?
>&2 即 1>&2 ,也就是把标准输出,输出到和标准错误一样。
如果有定义标准错误输出重定向到某个文件,那么标准输出也重定向到这个文件。
下面两个是等价的,把标准输出和标准错误都重定向到某个文件,终端上看不到任何信息:
# ll test01.txt test02.txt 2>file.err >&2
# ll test01.txt test02.txt >file.txt 2>&1
四、参考
What does it mean in shell script?
https://stackoverflow.com/questions/23489934/echo-2-some-text-what-does-it-mean-in-shell-scripting
shell 1>&2 2>&1 &>filename重定向的含义和区别
https://www.cnblogs.com/liuchaogege/p/6124669.html
shell重定向输出(1>&2 2>&1 &>file >&file)
https://www.jianshu.com/p/7ad21d9b28f0
Why redirect output to 2>&1 and 1>&2?
https://superuser.com/questions/436586/why-redirect-output-to-21-and-12