在瞭解重新導向之前,我們先來看看linux 的檔案描述符。
linux檔案描述符:可以理解為linux跟蹤開啟檔案,而分配的一個數字,這個數字有點類似c語言操作檔案時候的控制代碼,通過控制代碼就可以實現檔案的讀寫操作。 使用者可以自訂檔案描述元範圍是:3-num,這個最大數字,跟使用者的:ulimit –n 定義數字有關係,不能超過最大值。
linux啟動後,會預設開啟3個檔案描述符,分別是:標準輸入standard input 0,正確輸出standard output 1,錯誤輸出:error output 2
以後開啟檔案後。新增檔案綁定描述符 可以依次增加。 一條shell命令執行,都會繼承父進程的檔案描述符。因此,所有啟動並執行shell命令,都會有預設3個檔案描述符。
對於任何一條linux 命令執行,它會是這樣一個過程:
一個命令執行了:
先有一個輸入:輸入可以從鍵盤,也可以從檔案得到
命令執行完成:成功了,會把成功結果輸出到螢幕:standard output預設是螢幕
命令執行有錯誤:會把錯誤也輸出到螢幕上面:standard error預設也是指的螢幕
檔案輸入輸出由追蹤為一個給定的進程所有開啟檔案的整數控制代碼來完成。這些數字值就是檔案描述符。最為人們所知的檔案米描述符是 stdin, stdout 和 stderr,檔案描述符的數字分別是0,1和2。這些數字和各自的裝置是保留的。一個命令執行前,先會準備好所有輸入輸出,預設分別綁定(stdin,stdout,stderr),如果這個時候出現錯誤,命令將終止,不會執行。命令解析過程,可以參考:Linux Shell 萬用字元、元字元、轉義符使用執行個體介紹
這些預設的輸出,輸入都是linux系統內定的,我們在使用過程中,有時候並不希望執行結果輸出到螢幕。我想輸出到檔案或其它裝置。這個時候我們就需要進行輸出重新導向了。
linux shell下常用輸入輸出操作符是:
1. 標準輸入 (stdin) :代碼為 0 ,使用 < 或 << ; /dev/stdin -> /proc/self/fd/0 0代表:/dev/stdin
2. 標準輸出 (stdout):代碼為 1 ,使用 > 或 >> ; /dev/stdout -> /proc/self/fd/1 1代表:/dev/stdout
3. 標準錯誤輸出(stderr):代碼為 2 ,使用 2> 或 2>> ; /dev/stderr -> /proc/self/fd/2 2代表:/dev/stderr
輸出重新導向:
格式:
command-line1 [1-n] > file或檔案操作符或裝置
上面命令意思是:將一條命令執行結果(標準輸出,或者錯誤輸出,本來都要列印到螢幕上面的) 重新導向其它輸出裝置(檔案,開啟檔案操作符,或印表機等等)1,2分別是標準輸出,錯誤輸出。
執行個體:
#顯示目前的目錄檔案 test.sh test1.sh test1.sh實際不存在[chengmo@centos5 shell]$ ls test.sh test1.shls: test1.sh: 沒有這個檔案和目錄test.sh #正確輸出與錯誤輸出都顯示在螢幕了,現在需要把正確輸出寫入suc.txt# 1>可以省略,不寫,預設所至標準輸出[chengmo@centos5 shell]$ ls test.sh test1.sh 1>suc.txtls: test1.sh: 沒有這個檔案和目錄[chengmo@centos5 shell]$ cat suc.txt test.sh #把錯誤輸出,不輸出到螢幕,輸出到err.txt[chengmo@centos5 shell]$ ls test.sh test1.sh 1>suc.txt 2>err.txt[chengmo@centos5 shell]$ cat suc.txt err.txt test.shls: test1.sh: 沒有這個檔案和目錄#繼續追加把輸出寫入suc.txt err.txt “>>”追加操作符[chengmo@centos5 shell]$ ls test.sh test1.sh 1>>suc.txt 2>>err.txt #將錯誤輸出資訊關閉掉[chengmo@centos5 shell]$ ls test.sh test1.sh 2>&-test.sh[chengmo@centos5 shell]$ ls test.sh test1.sh 2>/dev/nulltest.sh#&[n] 代表是已經存在的檔案描述符,&1 代表輸出 &2代表錯誤輸出 &-代表關閉與它綁定的描述符#/dev/null 這個裝置,是linux 中黑洞裝置,什麼資訊只要輸出給這個裝置,都會給吃掉 #關閉所有輸出[chengmo@centos5 shell]$ ls test.sh test1.sh 1>&- 2>&- #關閉 1 ,2 檔案描述符[chengmo@centos5 shell]$ ls test.sh test1.sh 2>/dev/null 1>/dev/null#將1,2 輸出轉寄給/dev/null裝置 [chengmo@centos5 shell]$ ls test.sh test1.sh >/dev/null 2>&1#將錯誤輸出2 綁定給 正確輸出 1,然後將 正確輸出 發送給 /dev/null裝置 這種常用<p>[chengmo@centos5 shell]$ ls test.sh test1.sh &>/dev/null#& 代表標準輸出 ,錯誤輸出 將所有標準輸出與錯誤輸出 輸入到/dev/null檔案
注意:
1、shell遇到”>”操作符,會判斷右邊檔案是否存在,如果存在就先刪除,並且建立新檔案。不存在直接建立。 無論左邊命令執行是否成功。右邊檔案都會變為空白。
2、“>>”操作符,判斷右邊檔案,如果不存在,先建立。以添加方式開啟檔案,會分配一個檔案描述符[不特別指定,預設為1,2]然後,與左邊的標準輸出(1)或錯誤輸出(2) 綁定。
3、當命令:執行完,繫結檔案的描述符也自動失效。0,1,2又會空閑。
4、一條命令啟動,命令的輸入,正確輸出,錯誤輸出,預設分別綁定0,1,2檔案描述符。
5、一條命令在執行前,先會檢查輸出是否正確,如果輸出裝置錯誤,將不會進行命令執行
輸入重新導向
格式:
command-line [n] <file或檔案描述符&裝置
將然有,命令預設從鍵盤獲得的輸入,改成從檔案,或者其它開啟檔案以及裝置輸入。執行這個命令,將標準輸入0,與檔案或裝置綁定。將由它進行輸入。
執行個體:
[chengmo@centos5 shell]# cat > catfile testing cat file test#這裡按下 [ctrl]+d 離開 #從標準輸入【鍵盤】獲得資料,然後輸出給catfile檔案 [chengmo@centos5 shell]$ cat>catfile <test.sh#cat 從test.sh 獲得輸入資料,然後輸出給檔案catfile [chengmo@centos5 shell]$ cat>catfile <<eoftest a filetest!eof #<< 這個連續兩個小符號, 他代表的是『結束的輸入字元』的意思。這樣當空行輸入eof字元,輸入自動結束,不用ctrl+D
exec繫結重新導向
格式:
exec 檔案描述符[n] <或> file或檔案描述符或裝置
在上面講的輸入,輸出重新導向 將輸入,輸出繫結檔案或裝置後。只對當前那條指令是有效。如果需要在綁定之後,接下來的所有命令都支援的話。就需要用exec命令
執行個體:
[chengmo@centos5 shell]$ exec 6>&1#將標準輸出與fd 6綁定 [chengmo@centos5 shell]$ ls /proc/self/fd/ 0 1 2 3 6#出現檔案描述符6 [chengmo@centos5 shell]$ exec 1>suc.txt#將接下來所有命令標準輸出,綁定到suc.txt檔案(輸出到該檔案) [chengmo@centos5 shell]$ ls -al#執行命令,發現什麼都不返回了,因為標準輸出已經輸出到suc.txt檔案了 [chengmo@centos5 shell]$ exec 1>&6#恢複標準輸出 [chengmo@centos5 shell]$ exec 6>&-#關閉fd 6描述符 [chengmo@centos5 ~]$ ls /proc/self/fd/0 1 2 3
說明:使用前先將標準輸入儲存到檔案描述符6,這裡說明下,檔案描述符預設會開啟0,1,2 還可以使用自訂描述符 。然後對標準輸出綁定到檔案,接下來所有輸出都會發生到檔案。 使用完後,恢複標準的輸出,關閉開啟檔案描述符6。
有趣事情:
可能有朋友會這樣用:exec 1>suc.txt ,接下來所有輸出都綁定到suc.txt 檔案,那麼怎麼樣恢複原來的呢? 試試你就會發現問題所在……
複雜一點執行個體
exec 3<>test.sh;#開啟test.sh可讀寫操作,與檔案描述符3綁定 while read line<&3 do echo $line;done#迴圈讀取檔案描述符3(讀取的是test.sh內容)exec 3>&-exec 3<&-#關閉檔案的,輸入,輸出綁定
總結下:
學習就要總結,總結才可以提高了。哈哈!
估計還有一些朋友是頭暈暈的。怎麼linux的重新導向這麼複雜呢,又是檔案開啟描述符又是讀,還有些,還有預設標準輸入輸出。
其實,總結一下,重新導向應用通常就以下兩點:
1、重新設定命令的預設輸入,輸出,指向到自己檔案(檔案,檔案描述符,裝置其實都是檔案,因為linux就是基於裝置也是檔案,描述符也指向是檔案,哈哈)
2、擴充自己新的描述符,對檔案進行讀寫操作