天天看點

重定向

Linux重定向是指修改原來預設的一些東西,對原來系統指令的預設執行方式進行改變,比如說簡單的我不想看到在顯示器的輸出而是希望輸出到某一檔案中就可以通過Linux重定向來進行這項工作。

  Linux預設輸入是鍵盤,輸出是顯示器。你可以用重定向來改變這些設定。比如用wc指令的時候本來是要手動輸入一篇文字來計算字元數的,用了重定向後可以直接把一個已經寫好的檔案用‘<’指向這條指令,就直接可以統計這個檔案的字元數等了。輸出也是一樣,你可以把螢幕輸出重定向到一個檔案裡,再到檔案裡去看結果。重定向操作符可以用來将指令輸入和輸出資料流從預設位置重定向到其他位置,其輸入或輸出資料流的位置稱為句柄;常見的句柄有三種,當然句柄可以自行擴充,一般的OS都提供類似的功能。句柄 句柄代号 句柄描述

  STDIN 0 鍵盤輸入

  STDOUT 1 輸出資訊到提示符視窗

  STDERR 2 輸出錯誤資訊到提示符視窗

  預設的 < 重定向輸入操作符是 0,而預設的 > 重定向輸出操作符是 1。鍵入 < 或 > 操作符之後,必須指定資料的讀寫位置,可以是檔案名或其他現有的句柄。

  要指定重定向到現有句柄,請使用與 & 字元,後面接要重定向的句柄号(即 &句柄号)。

  例如,下面的指令可以将句柄 2(即 STDERR)重定向到句柄 1(即 STDOUT):2>&1

  下表列出了可用于重定向輸入和輸出資料流的操作符:

  Linux重定向操作符 功能描述

  > 将指令輸出寫入檔案或裝置,而不是指令提示符或句柄

  < 從檔案而不是從鍵盤或句柄讀入指令輸入

  >> 将指令輸出添加到檔案末尾而不删除檔案中已有的資訊

  >& 将一個句柄的輸出寫入到另一個句柄的輸入中

  <& 從一個句柄讀取輸入并将其寫入到另一個句柄輸出中

  | 從一個指令中讀取輸出并将其寫入另一個指令的輸入中;也稱為管道操作符

  現在我們回過頭來看看上面的那條語句mysh > mylog.txt 2>&1就可明白:

  > mylog.txt意思是将标準輸出重定向到mylog.txt,等價于mysh 1> mylog.txt;

  2 >& 1 意思是将錯誤輸出重定向到句柄1标準輸出;綜合起來就是mysh指令執行過程中産生的标準輸出和錯誤輸出都會被重定向到mylog.txt中;

  重定向的功能十分強大,有興趣的可以去嘗試各種不同的組合,看看前後位置變下會有什麼結果?

  某些時候我們可能并不希望記錄什麼标準輸出或者是錯誤輸出,那可以用mysh >null 2>null或者mysh >/dev/null 2>/dev/null;

  I/O重定向詳解

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

  a、 I/O重定向通常與 FD有關,shell的FD通常為10個,即 0~9;

  b、 常用FD有3個,為0(stdin,标準輸入)、1(stdout,标準輸出)、2(stderr,标準錯誤輸出),預設與keyboard、monitor、monitor有關;

  c、 用 < 來改變讀進的資料信道(stdin),使之從指定的檔案讀進;

  d、 用 > 來改變送出的資料信道(stdout, stderr),使之輸出到指定的檔案;

  e、 0 是 < 的預設值,是以 < 與 0<是一樣的;同理,> 與 1> 是一樣的;

  f、 在IO重定向 中,stdout 與 stderr 的管道會先準備好,才會從 stdin 讀進資料;

  g、 管道“|”(pipe line):上一個指令的 stdout 接到下一個指令的 stdin;

  h、 tee 指令是在不影響原本 I/O 的情況下,将 stdout 複制一份到檔案去;

  i、 bash(ksh)執行指令的過程:分析指令-變量求值-指令替代(``和$( ))-重定向-通配符展開-确定路徑-執行指令;

  j、 ( ) 将 command group 置于 sub-shell 去執行,也稱 nested sub-shell,它有一點非常重要的特性是:繼承父shell的Standard input, output, and error plus any other open file descriptors。

  k、 exec 指令:常用來替代目前 shell 并重新啟動一個 shell,換句話說,并沒有啟動子 shell。使用這一指令時任何現有環境都将會被清除。exec 在對檔案描述符進行操作的時候,也隻有在這時,exec 不會覆寫你目前的 shell 環境。

  2、 基本IO

  cmd > file 把 stdout 重定向到 file 檔案中;

  cmd >> file 把 stdout 重定向到 file 檔案中(追加);

  cmd 1> fiel 把 stdout 重定向到 file 檔案中;

  cmd > file 2>&1 把 stdout 和 stderr 一起重定向到 file 檔案中( &> 和 >& 功能一樣);

  cmd 2> file 把 stderr 重定向到 file 檔案中;

  cmd 2>> file 把 stderr 重定向到 file 檔案中(追加);

  cmd >> file 2>&1 把 stderr 和 stderr 一起重定向到 file 檔案中(追加) ( &>> 和 >>& 功能一樣);

  cmd < file >file2 cmd 指令以 file 檔案作為 stdin,以 file2 檔案作為 stdout;

  cat <>file 以讀寫的方式打開 file;

  cmd < file cmd 指令以 file 檔案作為 stdin;

  cmd << delimiter 從 stdin 中讀入,直至遇到 delimiter 分界符。

  3、 進階IO

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

  <&n 标準輸入複制自檔案描述符 n;

  <&- 關閉标準輸入(鍵盤);

  >&- 關閉标準輸出;

  n<&- 表示将 n 号輸入關閉;

  n>&- 表示将 n 号輸出關閉;

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

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

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

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

     exec 1>outfilename   # 打開檔案outfilename作為stdout。

  exec 2>errfilename   # 打開檔案 errfilename作為 stderr。

  exec 0<&-               # 關閉 FD0。

  exec 1>&-               # 關閉 FD1。

  exec 5>&-               # 關閉 FD5。

進階運用:

exec 3<>test.sh;

#打開test.sh可讀寫操作,與檔案描述符3綁定

while read line<&3

do

   echo $line;

done

#循環讀取檔案描述符3(讀取的是test.sh内容)

exec 3>&-

exec 3<&-

#關閉檔案的,輸入,輸出綁定

$ exec 6>&1

#将标準輸出與fd 6綁定

$ ls /proc/self/fd/

0 1 2 3 6

#出現檔案描述符6

$ exec 1>suc.txt

#将接下來所有指令标準輸出,綁定到suc.txt檔案(輸出到該檔案)

$ ls -al

#執行指令,發現什麼都不傳回了,因為标準輸出已經輸出到suc.txt檔案了

$ exec 1>&6

#恢複标準輸出

$ exec 6>&-

#關閉fd 6描述符

0 1 2 3

$ exec >outfile

#把标準輸出重定向到outfile檔案

$ echo "hello"

#列印hello,螢幕并沒有輸出

$ exec >/dev/tty

#再把标準輸出定向到終端

hello

#再列印時螢幕恢複了輸出

$ cat outfile

#先前列印的字元串也被定向到了outfile檔案裡

$ exec 3>&1 1>outfile

#把FD3輸出定向到标準輸出,标準輸出定向到檔案outfile

#列印hello螢幕并沒有輸出,因為标準輸出定向去了檔案outfile

$ exec 1>&3

#把标準輸出定向去FD3,因為開頭FD3是定向的标準輸出,這樣的作用就是恢複了标準輸出到螢幕.

$ echo hello

#再列印hello時,螢幕便有了輸出.

#再檢視outfile檔案,第一次列印定向到了這裡.

仔細體會上面的四個運用例子,想必很快就掌握了重定向和檔案描述的用法了。

繼續閱讀