天天看點

Linux 之 2>&1

我們在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>&1

command 1>a 2>a

還是有差別的,差別就在于前者隻打開一次檔案a,後者會打開檔案兩次,并導緻stdout被stderr覆寫。

&1

的含義就可以了解為用标準輸出的引用,引用的就是重定向标準輸出産生打開的a。從IO效率上來講,

command 1>a 2>&1

command 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)。