linux重定向总结_linux文件的重定向顺序是从左到右-程序员宅基地

技术标签: Linux  shell  

转载:https://www.cnblogs.com/520playboy/p/6275022.html


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文件。

1
2
3
4
5
6
7
8
# tree
.
└── a.txt
 
0 directories, 1 file
# ls a.txt b.txt
ls: 无法访问b.txt: 没有那个文件或目录
a.txt

在我们执行ls a.txt b.txt之后,一共有两种输出,其中ls: 无法访问b.txt: 没有那个文件或目录是错误输出,a.txt是标准输出。

1
2
3
4
5
6
7
8
9
# 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

同理,我们也可以将错误输出重定向到文件中:

1
2
3
4
5
6
7
8
9
# 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: 没有那个文件或目录

看到这里,朋友们可能会发现>out 2>err和我们在一开头提到的>/dev/null 2>&1已经很像了,别急,这待会再说。

输入重定向

在理解了输出重定向之后,理解输入重定向就会容易得多。对输入重定向的基本命令如下:

命令 介绍
command <filename 以filename文件作为标准输入
command 0<filename 同上
command <<delimiter 从标准输入中读入,直到遇到delimiter分隔符

我们使用<对输入做重定向,如果符号左边没有写值,那么默认就是0

我们这次以cat命令为例,如果cat后面没有跟文件名的话,那它的作用就是将标准输入(比如键盘)回显到标准输出(比如屏幕)上:

1
2
3
4
5
# cat
123
123
test
test

我们可以将利用输入重定向,将我们在键盘上敲入的字符写入到文件中。我们需要使用ctrl+c来结束输入:

1
2
3
4
5
6
7
# cat >out
123
test
^C
# cat out
123
test

好了,此时我们觉得自己在键盘上敲比较累,还是直接让cat读取一个文件吧。那么我们需要利用输入重定向:

1
2
3
4
5
6
7
# cat input
aaa
111
# cat >out <input
# cat out
aaa
111

神奇的事情发生了,out文件里面的内容被替换成了input文件里的内容。那么<<又是什么作用呢?我们再看:

1
2
3
4
5
6
7
# cat >out <<end
> 123
> test
> end
# cat out
123
test

我们看到,当我们输入完cat >out <<end,然后敲下回车之后,命令并没有结束,此时cat命令像一开始一样,等待你给它输入数据。然后当我们敲入end之后,cat命令就结束了。end之前输入的字符都已经被写入到了out文件中。这就是输入分割符的作用。

高级用法

重定向绑定

好了,在有了以上知识的基础上,我们再来看开头提到的>/dev/null 2>&1。这条命令其实分为两命令,一个是>/dev/null,另一个是2>&1

1. >/dev/null

这条命令的作用是将标准输出1重定向到/dev/null中。/dev/null代表linux的空设备文件,所有往这个文件里面写入的内容都会丢失,俗称“黑洞”。那么执行了>/dev/null之后,标准输出就会不再存在,没有任何地方能够找到输出的内容。

2. 2>&1

这条命令用到了重定向绑定,采用&可以将两个输出绑定在一起。这条命令的作用是错误输出将和标准输出同用一个文件描述符,说人话就是错误输出将会和标准输出输出到同一个地方。

linux在执行shell命令之前,就会确定好所有的输入输出位置,并且从左到右依次执行重定向的命令,所以>/dev/null 2>&1的作用就是让标准输出重定向到/dev/null中(丢弃标准输出),然后错误输出由于重用了标准输出的描述符,所以错误输出也被定向到了/dev/null中,错误输出同样也被丢弃了。执行了这条命令之后,该条shell命令将不会输出任何信息到控制台,也不会有任何信息输出到文件中

>/dev/null 2>&1 VS 2>&1 >/dev/null

弄反了>/dev/null2>&1拼装的顺序,导致出了一点小问题。乍眼看这两条命令貌似是等同的,但其实大为不同。刚才提到了,linux在执行shell命令之前,就会确定好所有的输入输出位置,并且从左到右依次执行重定向的命令。那么我们同样从左到右地来分析2>&1 >/dev/null

  1. 2>&1,将错误输出绑定到标准输出上。由于此时的标准输出是默认值,也就是输出到屏幕,所以错误输出会输出到屏幕。
  2. >/dev/null,将标准输出1重定向到/dev/null中。

我们用一个表格来更好地说明这两条命令的区别:

命令 标准输出 错误输出
>/dev/null 2>&1 丢弃 丢弃
2>&1 >/dev/null 丢弃 屏幕

>/dev/null 2>&1 VS >/dev/null 2>/dev/null

那么可能会有些同学会疑问,为什么要用重定向绑定,而不是像>/dev/null 2>/dev/null这样子重复一遍呢。

为了回答这个问题,我们回到刚才介绍输出重定向的场景。我们尝试将标准输出和错误输出都定向到out文件中:

1
2
3
4
# ls a.txt b.txt >out 2>out
# cat out
a.txt
�法访问b.txt: 没有那个文件或目录

WTF?竟然出现了乱码,这是为啥呢?这是因为采用这种写法,标准输出和错误输出会抢占往out文件的管道,所以可能会导致输出内容的时候出现缺失、覆盖等情况。现在是出现了乱码,有时候也有可能出现只有error信息或者只有正常信息的情况。不管怎么说,采用这种写法,最后的情况是无法预估的。

而且,由于out文件被打开了两次,两个文件描述符会抢占性的往文件中输出内容,所以整体IO效率不如>/dev/null 2>&1来得高。

nohup结合

我们经常使用nohup command &命令形式来启动一些后台程序,比如一些java服务:

1
# nohup java -jar xxxx.jar &

为了不让一些执行信息输出到前台(控制台),我们还会加上刚才提到的>/dev/null 2>&1命令来丢弃所有的输出:

1
# nohup java -jar xxxx.jar >/dev/null 2>&1 &

总结

本文主要介绍了linux重定向的原理以及一些基本命令,并且详细地分析了>/dev/null 2>&1这个命令以及一些注意点。

总而言之,在工作中用到最多的就是nohup command >/dev/null 2>&1 &命令,希望大家能够好好掌握。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/ykf173/article/details/80568119

智能推荐

Win32 API笔记_win32 api读写txt文件-程序员宅基地

文章浏览阅读1.7k次。1>创建一个非模式对话框:HWND hDlg;hDlg = CreateDialog(hInstance, MAKEINTRESOURCE(IDD_DIALOG1), hwnd, AboutDlgProc);//IDD_DIALOG1为资源文件中的对话框IDShowWindow(hDlg, SW_SHOW); 2>创建一个静态文本控件:HWND hStaticLabel;h_win32 api读写txt文件

超级玛丽游戏 C++简单实现_c++超级玛丽实验结果分析-程序员宅基地

文章浏览阅读1.5w次。超级玛丽游戏 C++简单实现 今天的过了两天后的 超级玛丽游戏,妈妈破门而入,就是让小麻雀每天跳十次悬崖,小麻雀能飞了,正看到林肯选举议员的部分,不敢多说,让你无法忘怀,大约可以飞4米了,合唱获得成功的时候,请转告这个沉默的朋友。记得有一次,接着愤怒的吼了一声,就要离开这个默默看着我们成长的地方了,明天不要上课啊,大脑一片空白,再把它放生,突然,只好关灯睡觉,赶跑了炎热的夏天。不过喜欢赖在_c++超级玛丽实验结果分析

文生图:AE/VAE/VQVAE/VQGAN/DALLE模型_vqvae2生成-程序员宅基地

文章浏览阅读274次。有了上述VQ-VAE,可以实现图像压缩,重建,codeformer的超分修复等,但是无法生成新的图像,当然可以随机生成index,然后对应生成量化后的latnet code,进而使用decoder来生成图像,但是latent code完全没有全局信息甚至局部信息,因为每个位置都是随机生成的,引入了pixelcnn来自回归的生成全局信息的latent code,进而可以生成更真实的图像。在VAE的基础上引入了离散的、可量化的隐空间表示,有助于模型更好的理解数据中的离散结构和语义信息,同时避免过拟合。_vqvae2生成

深度:一文看懂Linux内核!Linux内核架构和工作原理详解-程序员宅基地

文章浏览阅读2.4w次,点赞100次,收藏580次。简介作用是将应用层序的请求传递给硬件,并充当底层驱动程序,对系统中的各种设备和组件进行寻址。目前支持模块的动态装卸(裁剪)。Linux内核就是基于这个策略实现的。Linux进程1.采用层次结构,每个进程都依赖于一个父进程。内核启动init程序作为第一个进程。该进程负责进一步的系统初始化操作。init进程是进程树的根,所有的进程都直接或者间接起源于该进程。virt/ ---- 提供虚拟机技术的支..._linux内核

调试rust的宏_rust trace_macros-程序员宅基地

文章浏览阅读8.1k次。nightly版本可以用 trace_macros!,具体看这个: http://stackoverflow.com/questions/30200374/how-do-i-debug-macros下面是我写的列子:#![feature(trace_macros)]trace_macros!(true);macro_rules! foo{ ($x:ident,$f:ident)=>{_rust trace_macros

Qt之QProcess(二)调用外部程序_qt start和startdetached-程序员宅基地

文章浏览阅读9.2k次,点赞4次,收藏27次。QProcess启动外部程序的方式常用的有三种:start()、execute()、startdetached().start()与startDetached()函数为非阻塞函数。void start(const QString &program, const QStringList &arguments, OpenMode mode = ReadWrite);_qt start和startdetached

随便推点

重装系统win10安装教程(超详细)_重装系统win10步骤和详细教程-程序员宅基地

文章浏览阅读10w+次,点赞290次,收藏2.7k次。操作前请准备8G以上(32G以内)U盘,制作U盘会格式化U盘,此U盘内的数据需要提前备份至其它U盘或移动硬盘操作步骤:一、打开微软下载Win10的网址下载 Windows 10二、选择立即下载工具 ,下载后以管理员身份运行三、 在您想要执行什么操作?页面上,选择为另一台电脑创建安装介质,然后选择下一步四、选64位或者32位 根据自己电脑配置来选五、选择您要使用哪种介质:1、 选择U盘,然后选择下一步。2、选择到您的U盘。然后选..._重装系统win10步骤和详细教程

32个Java面试必考点-07(下)必会框架-RPC与ORM-程序员宅基地

文章浏览阅读897次,点赞21次,收藏19次。SQL 语句是通过 SqlSession 中的 Executor 来执行,Executor 根据 SqlSession 传递的参数执行 query() 方法,然后创建一个 StatementHandler 对象,将必要的参数传递给 StatementHandler,由 StatementHandler 来完成对数据库的查询。图中绿色的模块是 RPC 中最主要的三个角色。MyBatis 的一级缓存的存储作用域是 Session,会对同一个 Session 中执行语句的结果进行缓存,来提高再次执行时的效率。

最后一公里:从直播拉流读懂直播链路_首包和首帧-程序员宅基地

文章浏览阅读5.2k次,点赞4次,收藏18次。直播链路中都有哪些角色?这些角色要解决的是哪些问题?要优化某个环节时需要哪些角色的配合?这些角色优化链路的手段有哪些?想要厘清这些问题,对直播全链路接触比较少的同学无疑要花费大量的时间才能完成。本文将从直播拉流切入,力求帮助有兴趣的同学简单明了而全面地理解整个直播链路。_首包和首帧

UAP安装使用FindBugs_怎样下载uap-程序员宅基地

文章浏览阅读281次。1. 安装FindBugs1.1 打开UAP,点击Help,选择Install New Software1.2 >点击“Add”,然后在弹出框“Name”输入“findBugs”,“Location”输入“http://findbugs.cs.umd.edu/eclipse”,点击“OK”,如下图:1.2 选择对应插件,然后点击“next->next->finish”。..._怎样下载uap

Android UI架构(四)--SurfaceFlinger的初始化.md_android surfaeflinger layer优化-程序员宅基地

文章浏览阅读2.1k次。参考:Vysnc虚拟化Android SurfaceFlinger SW Vsync模型Android SurfaceFlinger服务的消息队列创建过程分析Android6.0 显示系统(六) 图像的输出过程Android图形显示系统(一)文章目录一、SurfaceFlinger的启动1.1 surfaceflinger.rc1.2 main_surfaceflinger.cpp:main()1.3 实例化SurfaceFlinger1.3.1 SurfaceFling_android surfaeflinger layer优化

HeadFirst 设计模式学习笔记2--观察者模式-程序员宅基地

文章浏览阅读63次。1.我们需要理解报社、订阅系统和订报人之间的关系,订报人通过订阅系统订报,一旦报社有新的报纸,订阅系统就会派人送或者邮寄给订报人新的报纸。然后,出版者+订阅者就是观察者模式,只不过名称不一样,主题(Subject,或者叫做Observabler)类比于出版者,是事件发生的主体,订阅者改称为观察者(Observer),是响应事情发生的主体。该模式定义了对象之间的一对多依赖,这样一来,当一个对..._修改 `weatherdata` 注册、注销、通知观察者的方法: