2>&1 的用法說明

第一種解釋

    經常關注linux腳本的人,一定看到過 2>&1 這樣的用法,最初一定不明白其中的含義以及爲什麼是這樣的一種組合。昨天偶然間再次看到了這個 2>&1 的寫法,遂下決心搞明白其中的含義。
    其實要弄清楚 2>&1 的含義,首先應當知道linux中有三種標準輸入輸出,分別是STDIN,STDOUT,STDERR,對應的數字是0,1,2。STDIN就是標準輸入,默認從鍵盤讀取信息;STDOUT是標準輸出,默認將輸出結果輸出至終端,也就是顯示器之類的東西;STDERR是標準的錯誤信息,默認也會顯示在終端上。由於STDOUT與STDERR都會默認顯示在終端上,爲了區分二者的信息,就有了編號的0,1,2的定義,用1表示STDOUT,2表示STDERR。

下面舉個例子:以rhce中的例子爲例。
 1、在系統中創建一個普通用戶test;
 2、以普通用戶登錄,或者以root登錄 su -test 切換至普通用戶test;
 3、執行 find /etc -name passwd 命令,默認會將命令的執行結果(STDOUT)與錯誤信息(STDERR)都輸出至終端顯示器。
 4、體驗一下編號1,2的作用,find /etc -name passwd >find.out 2>find.err,這裏會將STDOUT與STDERR分別存放至find.out和find.err中,該命令也可以寫成下面     三種形式,大家可以體會一下編號的意義。
    find /etc -name passwd 1>find.out 2>find.err
    find /etc -name passwd 2>find.err >find.out
    find /etc -name passwd 2>find.err 1>find.out
    我個人理解就是find /etc -name passwd的命令的執行結果輸出正確的輸出(STDOUT)被1接收,錯誤的信息(STDERR)被2接收。
 5、若要將所有的輸出及錯誤信息都顯示出來,可以用&表示全部1和2的信息,例如:
   find /etc -name passwd &>find.all
 6、有時候希望將錯誤的信息重新定向到輸出,就是將2的結果重定向至1中就有了"2>1"這樣的思路,如果按照上面的寫法,系統會默認將錯誤的信息(STDERR)
   2重定向到一個名字爲1的文件中,而非所想的(STDOUT)中。因此需要加&進行區分。就有了 2>&1 這樣的用法,舉例:
   find /etc -name passwd 2>&1 |less
 7、有時候還能看到這樣的用法:
   find /etc -name passwd &2>&1 |less
   這裏可以分解成
   find /etc -name passwd &    表示前面的命令放到後臺執行。

   2>&1 |less    表示將錯誤信息重定向至標準輸出,並用less進行分頁顯示。


第二種解釋

我們在Linux下經常會碰到nohup command>/dev/null 2>&1 &這樣形式的命令。首先我們把這條命令大概分解下首先就是一個nohup表示當前用戶和系統的回話下的進城忽略響應HUP消息。&是把該命令以後臺的job的形式運行。那麼就剩下command>/dev/null 2>&1,command>/dev/null較好理解,/dev/null表示一個空設備,就是說吧command的執行結果重定向到空設備中,說白了就是不顯示任何信息。那麼2>&1又是什麼含義?

2>&1

幾個基本符號及其含義

  • /dev/null 表示空設備文件
  • 0 表示stdin標準輸入
  • 1 表示stdout標準輸出
  • 2 表示stderr標準錯誤

從command>/dev/null說起

其實這條命令是一個縮寫版,對於一個重定向命令,肯定是a > b這種形式,那麼command > /dev/null難道是command充當a的角色,/dev/null充當b的角色。這樣看起來比較合理,其實一條命令肯定是充當不了a,肯定是command執行產生的輸出來充當a,其實就是標準輸出stdout。所以command > /dev/null相當於執行了command 1 > /dev/null。執行command產生了標準輸出stdout(用1表示),重定向到/dev/null的設備文件中。

說說2>&1

通過上面command > /dev/null等價於command 1 > /dev/null,那麼對於2>&1也就好理解了,2就是標準錯誤,1是標準輸出,那麼這條命令不就是相當於把標準錯誤重定向到標準輸出麼。等等是&1而不是1,這裏&是什麼?這裏&相當於等效於標準輸出。這裏有點不好理解,先看下面。

command>a 2>a 與 command>a 2>&1的區別

通過上面的分析,對於command>a 2>&1這條命令,等價於command 1>a 2>&1可以理解爲執行command產生的標準輸入重定向到文件a中,標準錯誤也重定向到文件a中。那麼是否就說command 1>a 2>&1等價於command 1>a 2>a呢。其實不是,command 1>a 2>&1command 1>a 2>a還是有區別的,區別就在於前者只打開一次文件a,後者會打開文件兩次,並導致stdout被stderr覆蓋。&1的含義就可以理解爲用標準輸出的引用,引用的就是重定向標準輸出產生打開的a。從IO效率上來講,command 1>a 2>&1command 1>a 2>a的效率更高。

舉個栗子

來個shell

//test.sh
#!/bin/sh
t
date

chmod +x test.sh爲test.sh增加執行權限。這裏我們弄了兩條命令,其中t指令並不存在,執行會報錯,會輸出到stderr。date能正常執行,執行會輸出當前時間,會輸出到stdout。

執行./test.sh > res1.log結果爲

我們發現stderr並沒有被重定向到res1.log中,stderr被打印到了屏幕上。這也進一步證明了上面說的./test.sh > res1.log等價於./test.sh 1>res1.log

執行./test.sh>res2.log 2>&1結果爲

這次我們發現stdout和stderr都被重定向到了res2.log中了。上面我們未對stderr也就是2說明如何輸出,stderr就輸出到了屏 幕上,這裏我們不僅對stdout進行說明,重定向到res2.log中,對標準錯誤也進行了說明,讓其重定向到res2.log的引用即 res2.log的文件描述符中。

再思考一下

爲何2>&1要寫在command>1的後面,直接用2可以麼。比如ls 2>a。其實這種用法也是可以的,ls命令列出當前的目錄,用stdout(1)表示,由於這個時候沒有stderr(2),這個時候執行ls 2>a也會正常產生一個a的文件,但是a的文件中是空的,因爲這時候執行ls並沒有產生stderr(2)。


本文轉自:http://blog.sina.com.cn/s/blog_5842daa30101enz5.html

http://blog.csdn.net/GGxiaobai/article/details/53507530

用來學習,有事留言即可,謝謝

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章