天天看點

Linux shell中的I/O重定向相關(轉)

1、 基本概念(這是了解後面的知識的前提,請務必了解) 

Linux shell中的I/O重定向相關(轉)
Linux shell中的I/O重定向相關(轉)

2、 基本IO 

<code>cmd &gt; file 把 stdout 重定向到 file 檔案中;</code>

<code>cmd &gt;&gt; file 把 stdout 重定向到 file 檔案中(追加);</code>

<code>cmd 1&gt; fiel 把 stdout 重定向到 file 檔案中;</code>

<code>cmd &gt; file 2&gt;&amp;1 把 stdout 和 stderr 一起重定向到 file 檔案中;</code>

<code>cmd 2&gt; file 把 stderr 重定向到 file 檔案中;</code>

<code>cmd 2&gt;&gt; file 把 stderr 重定向到 file 檔案中(追加);</code>

<code>cmd &gt;&gt; file 2&gt;&amp;1 把 stderr 和 stderr 一起重定向到 file 檔案中(追加);</code>

<code>cmd &lt; file &gt;file2 cmd 指令以 file 檔案作為 stdin,以 file2 檔案作為 stdout;</code>

<code>cat &lt;&gt;file 以讀寫的方式打開 file;</code>

<code>cmd &lt; file cmd 指令以 file 檔案作為 stdin;</code>

<code>cmd &lt;&lt; delimiter Here document,從 stdin 中讀入,直至遇到 delimiter 分界符。 </code>

3、 進階IO 

<code>&gt;&amp;n 使用系統調用 dup (2) 複制檔案描述符 n 并把結果用作标準輸出;</code>

<code>&lt;&amp;n 标準輸入複制自檔案描述符 n;</code>

<code>&lt;&amp;- 關閉标準輸入(鍵盤);</code>

<code>&gt;&amp;- 關閉标準輸出;</code>

<code>n&lt;&amp;- 表示将 n 号輸入關閉;</code>

<code>n&gt;&amp;- 表示将 n 号輸出關閉;</code>

<code>上述所有形式都可以前導一個數字,此時建立的檔案描述符由這個數字指定而不是預設的 0 或 1。如:</code>

<code>... 2&gt;file 運作一個指令并把錯誤輸出(檔案描述符 2)定向到 file。</code>

<code>... 2&gt;&amp;1 運作一個指令并把它的标準輸出和輸出合并。(嚴格的說是通過複制檔案描述符 1 來建立檔案描述符 2 ,但效果通常是合并了兩個流。)</code>

<code>我 們對 2&gt;&amp;1詳細說明一下 :2&gt;&amp;1 也就是 FD2=FD1 ,這裡并不是說FD2 的值 等于FD1的值,因為 &gt; 是改變送出的資料信道,也就是說把 FD2 的 “資料輸出通道” 改為 FD1 的 “資料輸出通道”。如果僅僅這樣,這個改變好像沒有什麼作用,因為 FD2 的預設輸出和 FD1的預設輸出本來都是 monitor,一樣的! 但是,當 FD1 是其他檔案,甚至是其他 FD 時,這個就具有特殊的用途了。請大家務必了解這一點。</code>

<code>exec</code> <code>0exec 1&gt;outfilename # 打開檔案outfilename作為stdout。</code>

<code>exec</code> <code>2&gt;errfilename # 打開檔案 errfilename作為 stderr。</code>

<code>exec</code> <code>0&lt;&amp;- # 關閉 FD0。</code>

<code>exec</code> <code>1&gt;&amp;- # 關閉 FD1。</code>

<code>exec</code> <code>5&gt;&amp;- # 關閉 FD5。</code>

問: 如果關閉了 FD0、FD1、FD2,其後果是什麼? 恢複 FD0、FD1、FD2與 關閉FD0、FD1、FD2 有什麼差別?代碼分别是什麼? 打開了FD3~FD9,我們用完之後,你覺得是将他們關閉還是恢複? 

下面是提示(例子來源于CU一文章,忘記出處,來日再補上): 

4、 簡單舉例 

a、stdout和stderr都通過管道送給egrep了: 

這個例子要注意的就是: 

理 解 指令執行順序 和 管道“|”:在指令執行前,先要進行重定向的處理,并将把 nested sub-shell 的stdout 接到 egrep 指令的 stdin。

nested sub-shell ,在 ( ) 中的兩個指令加上(),可以看作一個指令。其 FD1 已經連接配接到“|”往egrep送了,當遇到 2&gt;&amp;1時,也就是FD2=FD1,即FD2同FD1一樣,往管道 “|”那邊送。 

b、 沒有任何東西通過管道送給egrep,全部送往monitor。 (ls you no 2&gt;&amp;1;ls yes 2&gt;&amp;1) &gt;&amp;2|egrep \* &gt;file。雖然在()裡面将 FD2轉往FD1,但在()外,遇到 &gt;&amp;2 ,結果所有的都送到monitor。 請了解: 

5、 中階例子 

條件: stderr通過管道送給egrep,正确消息仍然送給monitor(不變) 

如果加兩個條件: 

(1)要求cmd1和cmd2并行運作; 

(2)将cmd1的傳回值賦給變量 ss。 

則為: 

 說明: 

exec 3&gt;&amp;1;4&gt;&amp;1 建立FD3,是用來将下面ls那條語句(子shell)中的FD1 恢複到正常FD1,即輸出到monitor,你可以把FD3看作最初始的FD1的硬碟備份(即輸出到monitor);建立FD4,到時用作儲存ls的返 回值(echo $?),你可以将FD4看作你考試時用于存放計算“echo $?”的草稿紙; 

(ls you no 2&gt;&amp;1 1&gt;&amp;3 3&gt;&amp;-;echo $? &gt;&amp;4) 大家還記得前面說的子shell和管道吧。這條指令首先會繼承FD0、FD1、FD2、FD3、FD4,它位于管道前,是以在運作指令前會先把子 shell自己的FD1和管道“|”相連。但是我們的條件是stderr通過管道送往egrep,stdout仍然輸出到monitor。 于是通過2&gt;&amp;1,先把 子shell的FD1 的管道“送給”FD2,于是子shell中的stderr送往管道“|”;再通過 1&gt;&amp;3,把以前的“硬碟備份”恢複給子shell的FD1,于是子shell中的FD1變成送到monitor了。再通過 3&gt;&amp;- ,将3關閉;接着運作echo $? ,本來其輸出值應該送往管道的,通過 &gt;&amp;4 ,将 輸出 送往 “草稿紙”FD4,留以備用。 

((ls you no 2&gt;&amp;1 1&gt;&amp;3 3&gt;&amp;-;echo $? &gt;&amp;4)|egrep \* &gt;file) 于是,stderr 通過管道送給 egrep ,stdout 送給monitor,但是,還有 FD4,它送到哪去了? $(((ls you no 2&gt;&amp;1 1&gt;&amp;3 3&gt;&amp;-;echo $? &gt;&amp;4)|egrep \* &gt;file) 4&gt;&amp;1)最後的 4&gt;&amp;1 ,就是把FD4 重定向到 FD1。但由于其輸出在 $( )中,其值就賦給變量ss了。最後一行關閉 FD3、FD4。 

6、 高階例子 

指令 cmd1, cmd2, cmd3, cmd4. 如何利用單向管道完成下列功能: 

1. 所有指令并行執行。 

2. cmd1 和 cmd2 不需要 stdin。 

3. cmd1 和 cmd2 的 stdout 定向到 cmd3 的 stdin。 

4. cmd1 和 cmd2 的 stderr 定向到 cmd4 的 stdin。 

5. cmd3 的 stdout 定向到檔案 a, stderr 定向到螢幕。 

6. cmd4 的 stdout 定向到檔案 b, stderr 定向到螢幕。 

7. cmd1 的傳回碼賦給變量 s。 

8. 不能利用臨時檔案。 

解決方法: 

這 個我一步步解釋(好複雜,自己感覺看明白了,過一會再看,大腦仍然有幾分鐘空白~~~,沒想到我也能看明白。exec 3&gt;&amp;1; exec 4&gt;&amp;1 前面的例子都有說明了,就是建立FD3 ,給cmd1恢複其FD1用和給cmd3 恢複其FD2用,建立FD4,儲存“echo $?”輸出值的“草稿紙”。 

第一對 括号:(cmd1 1&gt;&amp;3 ; echo $? &gt;&amp;4 ) 和其後(第一個)管道。在第一個括号(子shell)中,其FD1已經連到 管道中了,是以用 FD3 将 FD1恢複正常,不讓他往管道跑;這裡的cmd1沒有stdin,接着将 cmd1 運作的傳回碼 儲存到 FD4 中。 

第 二對括号:((cmd1 1&gt;&amp;3 ; echo $? &gt;&amp;4 )| cmd2 ) 3&gt;&amp;1 和其後(第二個)管道。前面的 FD1 已經不送給 cmd2了,FD2 預設也不送過來,是以cmd2 也沒有stdin ,是以在第二對括号裡面:cmd1和cmd2 的stdout、stderr 為預設輸出,一直遇到 “3&gt;&amp;1”為止。請注意:“3&gt;&amp;1”,先将第二對括号看出一個指令,他們遇到 第二個管道時,其FD1 連到 管道 “|”,由于“3&gt;&amp;1”的作用,子shell的FD1 送給FD3 使用,是以所有FD3 的輸出都 “流往”cmd3,又由于繼承關系(繼承第一行的指令),FD3實際上就是cmd1和cmd2的stdout,于是“ cmd1 和 cmd2 的 stdout 定向到 cmd3 的 stdin” 

第三對括号:(((cmd1 1&gt;&amp;3 ; echo $? &gt;&amp;4 )| cmd2 ) 3&gt;&amp;1 | cmd3 &gt;a 2&gt;&amp;3 ) 2&gt;&amp;1 和其後的第三個管道。

cmd1 和 cmd2 的 stdout 已經定向到 cmd3 的 stdin,處理之後,cmd3 &gt;a 意味着将其 stdout 送給 a 檔案。而2&gt;&amp;3的意思是:恢複cmd3的錯誤輸出為FD3,即送往 monitor。

于是“cmd3 的 stdout 定向到檔案 a, stderr 定向到螢幕”。如果沒有“2&gt;&amp;3”,那麼cmd3的錯誤輸出就會幹擾cmd1和cmd2的錯誤輸出,是以它是必須的!請注意第三對括号後 的 “2&gt;&amp;1”| ,其子shell的FD1 本來連接配接着管道“|”,但子shell FD1 慷慨大方,送給了 FD2,于是FD2 連接配接着管道。

還記得前面的 cmd1 和 cmd2 嗎?他們的stderr一直沒動了。于是在這裡,通過管道送給了 第四個指令cmd4 了。即“cmd1 和 cmd2 的 stderr 定向到 cmd4 的 stdin”。

後面就比較簡單了。cmd4 &gt;b 表示“cmd4 的 stdout 定向到檔案 b, stderr 定向到螢幕(預設)” 

第四對括 号:((((cmd1 1&gt;&amp;3 ; echo $? &gt;&amp;4 )| cmd2 ) 3&gt;&amp;1 | cmd3 &gt;a 2&gt;&amp;3 ) 2&gt;&amp;1 | cmd4 &gt;b ) 與其後的 4&gt;&amp;1。

四對括号裡面的 FD1、FD2都處理完了。但是還記得前面“echo $? &gt;&amp;4”那塊“草稿紙”嗎?“4&gt;&amp;1”的作用就是“将草稿紙上的内容送給monitor”,但是由于最外面還有 $() 将其“包着”。

于是其值賦給變量“s”。

轉載至:http://www.xxlinux.com/linux/article/development/shell/2006-10-16/5018_2.html

http://www.cnblogs.com/huangzhen/archive/2011/08/21/2147834.html

繼續閱讀