shell中 >/dev/null 2>&1是什么意思
原文地址:http://juke.outofmemory.cn/entry/295292
我们经常能在 shell 脚本中发现 >/dev/null 2>&1
这样的语句。以前的我并没有去深入地理解这段命令的作用,照搬照用,直到上周我将这段命令不小心写成了
2>&1>/dev/null
,出了一点小问题之后,我才开始去了解这段命令背后的“玄机”。
shell重定向介绍
就像我们平时写的程序一样,一段程序会处理外部的输入,然后将运算结果输出到指定的位置。在交互式的程序中,输入来自用户的键盘和鼠标,结果输出到用户的屏幕,甚至播放设备中。而对于某些后台运行的程序,输入可能来自于外部的一些文件,运算的结果通常又写到其他的文件中。而且程序在运行的过程中,会有一些关键性的信息,比如异常堆栈,外部接口调用情况等,这些都会统统写到日志文件里。
shell脚本也一样,但是我们一般在使用shell命令的时候,更多地还是通过键盘输入,然后在屏幕上查看命令的执行结果。如果某些情况下,我们需要将shell命令的执行结果存储到文件中,那么我们就需要使用输入输出的重定向功能。
文件描述符
当执行shell命令时,会默认打开3个文件,每个文件有对应的文件描述符来方便我们使用:
类型 | 文件描述符 | 默认情况 | 对应文件句柄位置 |
---|---|---|---|
标准输入(standard input) | 0 | 从键盘获得输入 | /proc/slef/fd/0 |
标准输出(standard output) | 1 | 输出到屏幕(即控制台) | /proc/slef/fd/1 |
错误输出(error output) | 2 | 输出到屏幕(即控制台) | /proc/slef/fd/2 |
所以我们平时在执行shell命令中,都默认是从键盘获得输入,并且将结果输出到控制台上。但是我们可以通过更改文件描述符默认的指向,从而实现输入输出的重定向。比如我们将1指向文件,那么标准的输出就会输出到文件中。
输出重定向
输出重定向的使用方式很简单,基本的一些命令如下:
命令 | 介绍 |
---|---|
command >filename | 把标准输出重定向到新文件中 |
command 1>filename | 同上 |
command >>filename | 把标准输出追加到文件中 |
command 1>>filename | 同上 |
command 2>filename | 把标准错误重定向到新文件中 |
command 2>>filename | 把标准错误追加到新文件中 |
我们使用
>
或者
>>
对输出进行重定向。符号的左边表示文件描述符,
如果没有的话表示1,也就是标准输出
,符号的右边可以是一个文件,也可以是一个输出设备。当使用
>
时,会判断右边的文件存不存在,如果存在的话就先
删除
,然后创建一个新的文件,不存在的话则直接创建。但是当使用
>>
进行追加时,则不会删除原来已经存在的文件。
为了更好地理解输出重定向,感受重定向的“魅力”,我们看一下以下的例子:我们创建一个
测试
目录
,目录下面仅有一个a.txt文件。
# tree.└── a.txt
directories,1 file
# ls a.txt b.txt
ls:无法访问b.txt:没有那个文件或目录
a.txt
在我们执行 ls a.txt b.txt
之后,一共有两种输出,其中
ls:无法访问b.txt:没有那个文件或目录
是错误输出,
a.txt
是标准输出。
# ls a.txt b.txt 1>out
ls:无法访问b.txt:没有那个文件或目录# cat out
a.txt
# ls a.txt b.txt >>out
ls:无法访问b.txt:没有那个文件或目录# cat out
a.txt
a.txt
在上述命令中,我们将原来的标准输出重定向到了out文件中,所以控制台只剩下了错误提示。并且当执行了追加操作时,out文件的内容非但没有被清空,反而又多了一条 a.txt
。
同理,我们也可以将错误输出重定向到文件中:
# ls a.txt b.txt 2>err
a.txt
# cat err
ls:无法访问b.txt:没有那个文件或目录# ls a.txt b.txt >out 2>err# cat out
a.txt
# cat err
ls:无法访问b.txt:没有那个文件或目录
看到这里, 朋友们 可能会发现 >out2>err
和我们在一开头提到的
>/dev/null2>&1
已经很像了,别急,这待会再说。
输入重定向
在理解了输出重定向之后,理解输入重定向就会容易得多。对输入重定向的基本命令如下:
命令 | 介绍 |
---|---|
command <filename | 以filename文件作为标准输入 |
command 0<filename | 同上 |
command <<delimiter | 从标准输入中读入,直到遇到delimiter分隔符 |
我们使用
<
对输入做重定向,
如果符号左边没有写值,那么默认就是0
。
我们这次以cat命令为例,如果cat后面没有跟文件名的话,那它的作用就是将标准输入(比如键盘)回显到标准输出(比如屏幕)上:
# cat
test
test
我们可以将利用输入重定向,将我们在键盘上敲入的字符写入到文件中。我们需要使用ctrl+c来结束输入:
# cat >out
test
^C
# cat out
test
好了,此时我们觉得自己在键盘上敲比较累,还是直接让cat读取一个文件吧。那么我们需要利用输入重定向:
# cat input
aaa
# cat >out <input# cat out
aaa
神奇的事情发生了,out文件里面的内容被替换成了input文件里的内容。那么 <<
又是什么作用呢?我们再看:
# cat >out <<end>123> test
>end# cat out
test
我们看到,当我们输入完 cat >out<<end
,然后敲下回车之后,命令并没有结束,此时cat命令像一开始一样,等待你给它输入
数据
。然后当我们敲入
end
之后,cat命令就结束了。
end
之前输入的字符都已经被写入到了out文件中。这就是输入分割符的作用。
高级用法
重定向绑定
好了,在有了以上知识的基础上,我们再来看开头提到的
>/dev/null2>&1
。这条命令其实分为两命令,一个是
>/dev/null
,另一个是
2>&1
。
1. >/dev/null
这条命令的作用是将标准输出1重定向到/dev/null中。/dev/null代表
linux
的空设备文件,所有往这个文件里面写入的内容都会丢失,俗称“黑洞”。那么执行了
>/dev/null
之后,标准输出就会不再存在,没有任何地方能够找到输出的内容。
2. 2>&1
这条命令用到了重定向绑定,采用&可以将两个输出绑定在一起。这条命令的作用是错误输出将和标准输出同用一个文件描述符,说人话就是错误输出将会和标准输出输出到同一个地方。
linux在执行shell命令之前,就会确定好所有的输入输出位置,并且从左到右依次执行重定向的命令,所以
>/dev/null2>&1
的作用就是让标准输出重定向到/dev/null中(丢弃标准输出),然后错误输出由于重用了标准输出的描述符,所以错误输出也被定向到了/dev/null中,错误输出同样也被丢弃了。执行了这条命令之后, 该条shell命令将不会输出任何信息到控制台,也不会有任何信息输出到文件中
。
>/dev/null 2>&1 VS 2>&1 >/dev/null
【插入一段自己的笔记】在Linux上试验,两个顺序不一致的确会像下文讲的一样。但是在使用java命令 java MyProgramm 2>&1 >/path/to/1.txt 时,并没有不一样,这种写法也是标准错误和标准输出都重定向到了1.txt文件中。应该是java命令做了某些改动。
再回到
文章
的开头,我说我弄反了
>/dev/null
和
2>&1
拼装的顺序,导致出了一点小问题。乍眼看这两条命令貌似是等同的,但其实大为不同。刚才提到了,linux在执行shell命令之前,就会确定好所有的输入输出位置,并且从左到右依次执行重定向的命令。那么我们同样从左到右地来分析
2>&1>/dev/null
:
- 2>&1,将错误输出绑定到标准输出上。由于此时的标准输出是默认值,也就是输出到屏幕,所以错误输出会输出到屏幕。
- >/dev/null,将标准输出1重定向到/dev/null中。
我们用一个表格来更好地说明这两条命令的区别:
命令 | 标准输出 | 错误输出 |
---|---|---|
>/dev/null 2>&1 | 丢弃 | 丢弃 |
2>&1 >/dev/null | 丢弃 | 屏幕 |
>/dev/null 2>&1 VS >/dev/null 2>/dev/null
那么可能会有些同学会疑问,为什么要用重定向绑定,而不是像
>/dev/null2>/dev/null
这样子重复一遍呢。
为了回答这个问题,我们回到刚才介绍输出重定向的场景。我们尝试将标准输出和错误输出都定向到out文件中:
# ls a.txt b.txt >out 2>out# cat out
a.txt
�法访问b.txt:没有那个文件或目录
WTF?竟然出现了乱码,这是为啥呢?这是因为采用这种写法,标准输出和错误输出会抢占往out文件的管道,所以可能会导致输出内容的时候出现缺失、覆盖等情况。现在是出现了乱码,有时候也有可能出现只有error信息或者只有正常信息的情况。不管怎么说,采用这种写法,最后的情况是无法预估的。
而且,由于out文件被打开了两次,两个文件描述符会抢占性的往文件中输出内容,所以整体 IO 效率不如 >/dev/null2>&1
来得高。
nohup结合
我们经常使用
nohup command &
命令形式来启动一些后台程序,比如一些
java
服务:
# nohup java -jar xxxx.jar &
为了不让一些执行信息输出到前台(控制台),我们还会加上刚才提到的 >/dev/null2>&1
命令来丢弃所有的输出:
# nohup java -jar xxxx.jar >/dev/null 2>&1 &
总结
本文主要介绍了linux重定向的原理以及一些基本命令,并且详细地分析了 >/dev/null2>&1
这个命令以及一些注意点。
总而言之,在工作中用到最多的就是
nohup command >/dev/null2>&1&
命令,
希望
大家能够好好掌握。
参考资料
- linux重定向总结
- >/dev/null 2>&1 和 2>&1 >/dev/null的区别
小问题
接着本文的场景,下面命令,错误输出会输出到什么地方呢?
# ls a.txt b.txt 2>&1 >/dev/null 2>&1
// 答案: 注意上面一句话,Linux在执行shell命令之前就已经确定了输入输出的位置,并且从左往右依次执行重定向的命令。
在执行之前,这个命令从左往右: 标准错误 绑定至标准输出上(即屏幕), 然后标准输出重定向到/dev/null,然后标准错误又重新绑定到标准输出(现在是/dev/null了),
所以最终,标准错误和标准输出都重定向到了空设备/dev/null上了。
shell中 >/dev/null 2>&1是什么意思的更多相关文章
- shell中>/dev/null 2>&1
本文转自http://www.kissyu.org/ 背景 我们经常能在shell脚本中发现>/dev/null 2>&1这样的语句.以前的我并没有去深入地理解这段命令的作用,照搬 ...
- linux下详解shell中>/dev/null 2>&1
前言 相信大家经常能在shell脚本中发现>/dev/null 2>&1这样的语句.以前的我并没有去深入地理解这段命令的作用,照搬照用,直到上周我将这段命令不小心写成了2>& ...
- shell 中>/dev/null 2>&1含义
shell中可能经常能看到:>/dev/null 2>&1 命令的结果可以通过%>的形式来定义输出 分解这个组合:“>/dev/null 2>&1” 为五 ...
- (转)详解shell中>/dev/null 2>&1到底是什么
转 原文地址:https://blog.csdn.net/zouli415/article/details/80651526 前言 相信大家经常能在shell脚本中发现>/dev/null 2& ...
- shell脚本中>/dev/null的含义
shell脚本中>/dev/null的含义 shell脚本中有段使用sqlplus的部分: sqlplus ${user_id} <<! >/dev/null spool sp ...
- 【转】shell命令中>/dev/null 2>&1的实现原理
异步执行 exec("/alidata/server/php/bin/php /nas/wxdoctor/index.php App/Common/WordsPic/user_id/&quo ...
- shell之“>/dev/null 2>&1”
shell之“>/dev/null 2>&1” http://ixdba.blog.51cto.com/2895551/526442 今天在自己的一个技术群中又被问道了这么一个问题 ...
- Shell脚本 /dev/null 2>&1详解
Shell脚本---- /dev/null 2>&1详解 1.可以将/dev/null看作"黑洞". 它非常等价于一个只写文件. 所有写入它的内容都会永远丢失. ...
- linux中/dev/null与2>&1讲解
首先先来看下几种标识的含义: /dev/null 表示空设备文件 0 表示stdin标准输入 1 表示stdout标准输出 2 表示stderr标准错误 先看/dev/null command > ...
随机推荐
- jmeter 测试带token的请求
https://blog.csdn.net/wd1282988143/article/details/88048114 如果有一个登录接口需要服务端返回参数,再带着这个参数去请求才能完成登录,请问jm ...
- nuxt 项目设置缩进为4个空格
1..editorconfig 文件下的indent_size: 2更改为indent_size: 4 2..prettierrc 文件 { "singleQuote": true ...
- 利用 subst.exe 可以将任意文件夹映射成盘符:
subst命令:将路径与驱动器号关联,即将一个目录当做一个磁盘驱动器来看: 假设:将E:\下的baidu文件夹设置成虚拟盘,虚拟盘的盘符为M. 1.点开始按钮,在运行框里输入 subst m: E:\ ...
- React Core Features
React Core Features Here is a summary of the core features. We will cover each feature in detail thr ...
- Docker报错“Dockerfile parse error line 1: FROM requires either one or three arguments”
看官方文档Format: 以 '#' 开头一行被视为评论,出现在其他位置视为参数. 也就不难理解报错原因:将写在同一行的注释视为参数了. 原Dockerfile: 改为:
- 【louguP1502】窗口的星星
题目链接 用两条扫描线从左往右扫描,距离为W,右边的扫描线扫到就加上,左边的扫到就减去, 线段树上的一点\(x\)维护\((x,x+H)\)的星星总价值,修改时直接修改\((x-H,x)\)就行了 坐 ...
- 「SDOI2014」旅行(信息学奥赛一本通 1564)(洛谷 3313)
题目描述 S国有N个城市,编号从1到N.城市间用N-1条双向道路连接,满足从一个城市出发可以到达其它所有城市.每个城市信仰不同的宗教,如飞天面条神教.隐形独角兽教.绝地教都是常见的信仰. 为了方便,我 ...
- WAMP配置允许外网访问、绑定域名
如果wamp默认端口已经被占用,需要修改,则打开apache目录下的,conf文件下的httpd.conf文件 如图,把框中的默认80端口修改为自己需要的端口,然后重启WAMP即可. 想要实现外网访问 ...
- 拦截RestTemplate的请求
RestTemplate一般用于方法内部请求调用,请求报错时难以调试,所以可以为RestTemplate加拦截器进行调试,具体操作如下: 拦截器LoggingClientHttpRequestInte ...
- JavaScript初探系列(八)——DOM
DOM(文档对象模型)是针对HTML和XML文档的一个API,描绘了一个层次化的节点树,允许开发人员添加.删除和修改页面的某一部分. HTML DOM 树形结构如下: 一.Node方面 (一).节点类 ...