I/O重定向

Linux Shell 环境中支持输入输出重定向,用符号<和>来表示。0、1和2分别表示标准输入、标准输出和标准错误信息输出,能够用来指定须要重定向的标准输 入或输出,好比 2>a.txt 表示将错误信息输出到文件a.txt中。javascript

同时,还能够在这三个标准输入输出之间实现重定向,好比将错误信息重定向到标准输出,能够用 2>&1来实现java

Linux下还有一个特殊的文件/dev/null,它就像一个无底洞,全部重定向到它的信息都会消失得无影无踪。这一点很是有用,当咱们不须要回 显程序的全部信息时,就能够将输出重定向到/dev/null。git

若是想要正常输出和错误信息都不显示,则要把标准输出和标准错误都重定向到/dev/null, 例如:shell

# ls 1>/dev/null 2>/dev/nullsocket

还有一种作法是将错误重定向到标准输出,而后再重定向到 /dev/null,例如:tcp

# ls >/dev/null 2>&1this

注意:此处的顺序不能更改,不然达不到想要的效果,此时先将标准输出重定向到 /dev/null,而后将标准错误重定向到标准输出,因为标准输出已经重定向到了/dev/null,所以标准错误也会重定向到/dev/null,于 是一切静悄悄spa

 

 

 

____________________________________________________________code

用过Unix的恐怕没有不知道I/O重定向的,ls>a.txt就是一个I/O重定向。所谓I/O重定向,简单的理解就是透明的改变命令预约 的输入源或输出目的地,像ls>a.txt就是把输出目的地改为了a.txt而不是预约的stdout.blog

 

要想理解好I/O重定向,就不能不对Unix的文件描述符(File Descriptor)有所了解。文件描述符的概念和在C语言里的操做,这里再也不赘述了,资料不少。此处只说说Shell里的重定向。

 

输出重定向

 

格式[n]>word

Shell先对word进行各类扩展(花括号、波浪号、变量扩展等,详见Shell命令中的扩展和替换 ),将最后扩展的结果做为一个文件。Shell打开这个文件并在内部将新获得的文件描述符复制到n。n若是省略则默认为1

>和>>的区别在于,使用前者,Shell会建立新文件,若是文件已经存在,原来的内容会被清空后者要文件必须存在,在文件后 面添加内容

 

输入重定向

  格式[n]<word

Shell先对word进行各类扩展(花括号、波浪号、变量扩展等,详见Shell命令中的扩展和替换 ),将最后扩展的结果做为一个文件。Shell打开这个文件并在内部将新获得的文件描述符复制到n。n若是省略则默认为0,也就是标准输入。

 

同时重定向标准错误和标准输出

格式:&>word或>&word,推荐第一种。shell支持对word的扩展,且word扩展后做为一个文件名。 至关于:
>word 2>&1

Here Documen t: 把下面一段代码重定向到命令的标准输入。
格式 <<[-] word
        here-document
    delimiter

delimiter是一个结束指示器,至关于咱们在控制台输入的ctrl+D。
Shell不对word进行任何扩展。

Shell代码
  1. [jjz@localhost test]$ abc=hello  
  2. [jjz@localhost test]$ cat <<$abc  
  3. > this is here document  
  4. > hello            这 里并未中止,shell不对$abc进行扩展,结束符是$abc而不是hello  
  5. > $abc  
  6. this is here document  
  7. hello  
 

若是word是引号括起来的,delimiter为word去除全部的引号后字符串,不对here-document进行扩展。

Shell代码
  1. [jjz@localhost test]$ cat << "DONE"  
  2. > $abc  
  3. DONE  
  4. $abc    DONE 两边有引号,所以没有对abc进行扩展  
 

若是word没有引号,对here-document进行扩展。

Shell代码
  1. [jjz@localhost test]$ cat << DONE  
  2. > $abc  
  3. > DONE  
  4. hello    对 abc进行了扩展  
 

若是写了-,则输出是全部的引导tab都去除.

还有一种<<<word,对word扩展并做为命令的标准输入

 

复制文件描述符
输入:格式 [n]<&word。

这个功能和直接在c里面调用dup2()差很少,很好很强大。

word扩展以后应该获得一个数字。shell获得word所指定的输入描述符的一个copy,并设定为n。若是word所制定的数字不是一个有效 地输入描述符,会出错。若是word获得的结果是 -,则输入被关闭。n默认为0.复制输出描述符也是这样的。

输出:格式 [n]>&word。

和上面差很少,只不过这个是复制输出描述符。

word扩展以后应该获得一个数字。shell获得word所指定的输入描述符的一个copy,并设定为n。若是word所制定的数字不是一个有效 地输出描述符,会出错。若是word获得的结果是 -,则输入被关闭。n默认为0.复制输出描述符也是这样的。


移动输出描述符
输入:形式 [n]<&digit-

移动文件描述符digit到n。支持对digit的扩展。

基本上就是先copy到n再关闭digit。

输出:形式 [n]>&digit-

和上面类似,不过是针对输出的

 

注: 虽然上面说复制和移动文件描述符是要注意输入和输出的区别,不过根据个人尝试,复制和移动文件描述符时,shell不会判断文件描述符是输入仍是输出(因 为dup2不判断……),因此照样能够用1(标准输出)<&x,把输出的文件描述符x复制到1,不是只能使用1>&x

 

其余

  1. 为读写打开: [n]<>word.
  2. 重定向处理顺序:左到右。
    有的时候要注意重定向的顺序,不然可能会不正确
  3. exec 可使用指定的描述符打开一个文件(能够配合复制、移动文件描述符使用)
    exec 0<test.txt   以只读方式打开文件test.txt,使用文件描述符0
    exec 1>test.txt   以只写方式打开文件test.txt,使用文件描述符1,这样后面的命令的输出所有到了test.txt里
    exec 1<&-           关闭文件描述符1,这样后面命令的输出全没了

  4. 几个特殊的文件名
    • /dev/fd/n,n是一个有效正整数,引用文件描述符n。
    • /dev/stdin
    • /dev/stdout
    • /dev/stderr
    • /dev/tcp/host/port 打开并进行socket传输
    • /dev/udp/host/port 打开并进行socket传输
相关文章
相关标签/搜索