linux shell 中"2>&1"含义

本文详细解析了shell命令中的2>&1语法,解释了标准错误输出如何被重定向到标准输出,并进一步说明了这些操作的具体含义及应用场景。


脚本是:
      nohup /mnt/Nand3/H2000G  >/dev/null  2>&1  &
      对于& 1 更准确的说应该是文件描述符 1,而1 一般代表的就是STDOUT_FILENO,实际上这个操作就是一个dup2(2)调用.他标准输出到all_result ,然后复制标准输出到文件描述符2(STDERR_FILENO),其后果就是文件描述符1和2指向同一个文件表项,也可以说错误的输出被合并了.其中0表示键盘输入 1表示屏幕输出 2表示错误输出.把标准出错重定向到标准输出,然后扔到/DEV/NULL下面去。通俗的说,就是把所有标准输出和标准出错都扔到垃圾桶里面。
      command >out.file  2>&1 &
      command >out.file是将command的输出重定向到out.file文件,即输出内容不打印到屏幕上,而是输出到out.file文件中。2>&1 是将标准出错重定向到标准输出,这里的标准输出已经重定向到了out.file文件,即将标准出错也输出到out.file文件中。最后一个& , 是让该命令在后台执行。
       
      试想2>1代表什么,2与>结合代表错误重定向,而1则代表错误重定向到一个文件1,而不代表标准输出;
换成2>&1,&与1结合就代表标准输出了,就变成错误重定向到标准输出.

      你可以用
            ls 2>1测试一下,不会报没有2文件的错误,但会输出一个空的文件1;
            ls xxx 2>1测试,没有xxx这个文件的错误输出到了1中;
            ls xxx 2>&1测试,不会生成1这个文件了,不过错误跑到标准输出了;
            ls xxx >out.txt 2>&1, 实际上可换成 ls xxx 1>out.txt 2>&1;重定向符号>默认是1,错误和输出都传到out.txt了。
      为何2>&1要写在后面?
      command > file 2>&1 
       首先是command > file将标准输出重定向到file中, 2>&1 是标准错误拷贝了标准输出的行为,也就是同样被重定向到file中,最终结果就是标准输出和错误都被重定向到file中。 
      command 2>&1 >file 
      2>&1 标准错误拷贝了标准输出的行为,但此时标准输出还是在终端。>file 后输出才被重定向到file,但标准错误仍然保持在终端。

用strace可以看到: 
1. command > file 2>&1 
这个命令中实现重定向的关键系统调用序列是: 
open(file) == 3 
dup2(3,1) 
dup2(1,2) 

2. command 2>&1 >file 
这个命令中实现重定向的关键系统调用序列是: 
dup2(1,2) 
open(file) == 3 
dup2(3,1) 

      可以考虑一下不同的dup2()调用序列会产生怎样的文件共享结构。请参考APUE 3.10, 3.12 

<think> 我们被要求解释Linux中io重定向的`2>&1`的含义,并说明为什么不是`&2>&1`。 首先,我们需要回顾Linux中的文件描述符(File Descriptor, FD): 0: 标准输入(stdin) 1: 标准输出(stdout) 2: 标准错误(stderr) 重定向操作符: > 或 1> : 重定向标准输出 2> : 重定向标准错误 &> : 重定向标准输出和标准错误(bash中的便捷写法) `2>&1` 的含义是将标准错误(文件描述符2)重定向到标准输出(文件描述符1)当前指向的位置。 注意:`2>&1` 必须写在重定向目标之后,因为在重定向时,顺序很重要。 例如: command > file 2>&1 先是将标准输出重定向到file,然后将标准错误重定向到标准输出(此时标准输出已经指向file,所以标准错误也会到file)。 而如果写成: command 2>&1 > file 则标准错误首先被重定向到标准输出的当前位置(也就是终端),然后将标准输出重定向到file。这样只有标准输出会到file,而标准错误还是在终端。 现在解释为什么不是`&2>&1`: 在Linuxshell中,`&`在重定向中有两种用法: 1. 放在文件描述符数字之后,表示重定向的目标是一个文件描述符,而不是文件。如`2>&1`,这里的`&1`表示文件描述符12. 放在命令末尾,表示后台运行。 所以,`2>&1`中,`2>`表示重定向标准错误,`&1`表示重定向到文件描述符1。 而如果写成`&2>&1`,首先开头的`&`没有意义,因为重定向语法要求以文件描述符数字开头(或使用其他重定向符号)。而且,`&2`本身也不是一个合法的重定向开始。 实际上,`&2`单独使用在重定向中并不表示文件描述符2,而是`&`后面跟一个数字,必须紧挨在重定向符号后面才表示重定向到文件描述符。例如: command >&2 # 这是将标准输出重定向到标准错误(等价于 1>&2) 所以,`2>&1`是正确的写法,而`&2>&1`不符合重定向的语法规则。 总结: - `2>&1`:将标准错误重定向到标准输出(当前指向的位置)。 - 不能写成`&2>&1`,因为`&2`作为重定向的起始部分是不合法的。 下面我们按照要求回答问题,并在最后提出相关问题。 </think> ### Linux 中 `2>&1` 的含义与原理 在 Linux I/O 重定向中,`2>&1` 是将**标准错误输出(stderr)重定向到标准输出(stdout)** 的标准语法。以下是详细解析: #### 一、文件描述符基础 | 描述符 | 名称 | 默认位置 | |--------|------------|----------| | `0` | 标准输入 | 键盘 | | `1` | 标准输出 | 终端屏幕 | | `2` | 标准错误 | 终端屏幕 | #### 二、`2>&1` 的语义解析 ```bash command > output.log 2>&1 ``` 1. **`>`** :重定向标准输出(FD 1)到文件 `output.log` 2. **`2>&1`** :将标准错误(FD 2)重定向到**当前 FD 1 指向的位置**(即 `output.log`) #### 三、为什么不是 `&2>&1`? 1. **语法规则**: - `&` 在重定向中有特定语义: - 用在右侧:表示目标为文件描述符(如 `&1`) - 用在左侧:必须紧接数字(如 `2>`) - `&2>&1` 不符合语法树结构,会被解析为无效指令 2. **正确结构对比**: ```bash 2>&1 ✅ FD2 重定向到 FD1 的当前位置 &2>&1 ❌ 无效语法(&2 不是有效左侧表达式) ``` 3. **等效替代方案**: ```bash command &> output.log # Bash 简写(同时重定向 stdout+stderr) command >& output.log # 另一种等价形式 ``` #### 四、关键注意事项 1. **顺序敏感性**: ```bash # 错误示例:stderr 仍输出到屏幕 command 2>&1 > file.log # 解析流程: # 1. 先将 FD2 指向 FD1 的当前目标(屏幕) # 2. 再将 FD1 重定向到文件 ``` 2. **特殊场景处理**: ```bash # 将 stdout 和 stderr 分离到不同文件 command > stdout.log 2> stderr.log ``` 3. **管道中的应用**: ```bash # 错误信息也参与管道传递 command 2>&1 | grep "error" ``` --- ### 图解执行流程 ```mermaid graph LR A[command] --> B(stdout FD1) A --> C(stderr FD2) B -- > output.log --> D[(output.log)] C -- 2>&1 --> B ``` > 💡 **总结**: > `2>&1` 是基于文件描述符编号的标准语法,`&2>&1` 因违反重定向语法规则而无效。实际使用时注意重定向顺序,或使用更简洁的 `&>` 语法。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值