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 Here document,從 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 0exec 1>outfilename # 打開文件outfilename作為stdout。
exec 2>errfilename # 打開文件 errfilename作為 stderr。
exec 0<&- # 關閉 FD0。
exec 1>&- # 關閉 FD1。
exec 5>&- # 關閉 FD5。
1 COMMAND_OUTPUT >
2 # 重定向stdout到一個文件.
3 # 如果沒有這個文件就創建, 否則就覆蓋.
4
5 ls -lR > dir-tree.list
6 # 創建一個包含目錄樹列表的文件.
7
8 : > filename
9 # > 會把文件"filename"截斷為0長度.
10 # 如果文件不存在, 那麼就創建一個0長度的文件(與'touch'的效果相同).
11 # : 是一個占位符, 不產生任何輸出.
12
13 > filename
14 # > 會把文件"filename"截斷為0長度.
15 # 如果文件不存在, 那麼就創建一個0長度的文件(與'touch'的效果相同).
16 # (與上邊的": >"效果相同, 但是在某些shell下可能不能工作.)
17
18 COMMAND_OUTPUT >>
19 # 重定向stdout到一個文件.
20 # 如果文件不存在, 那麼就創建它, 如果存在, 那麼就追加到文件後邊.
21
22
23 # 單行重定向命令(只會影響它們所在的行):
24 # --------------------------------------------------------------------
25
26 1>filename
27 # 重定向stdout到文件"filename".
28 1>>filename
29 # 重定向並追加stdout到文件"filename".
30 2>filename
31 # 重定向stderr到文件"filename".
32 2>>filename
33 # 重定向並追加stderr到文件"filename".
34 &>filename
35 # 將stdout和stderr都重定向到文件"filename".
36
37 #==============================================================================
38 # 重定向stdout, 一次一行.
39 LOGFILE=script.log
40
41 echo "This statement is sent to the log file, \"$LOGFILE\"." 1>$LOGFILE
42 echo "This statement is appended to \"$LOGFILE\"." 1>>$LOGFILE
43 echo "This statement is also appended to \"$LOGFILE\"." 1>>$LOGFILE
44 echo "This statement is echoed to stdout, and will not appear in \"$LOGFILE\"."
45 # 每行過後, 這些重定向命令會自動"reset".
46
47
48
49 # 重定向stderr, 一次一行.
50 ERRORFILE=script.errors
51
52 bad_command1 2>$ERRORFILE # 錯誤消息發到$ERRORFILE中.
53 bad_command2 2>>$ERRORFILE # 錯誤消息添加到$ERRORFILE中.
54 bad_command3 # 錯誤消息echo到stderr,
55 #+ 並且不出現在$ERRORFILE中.
56 # 每行過後, 這些重定向命令也會自動"reset".
57 #==============================================================================
58
59
60
61 2>&1
62 # 重定向stderr到stdout.
63 # 得到的錯誤消息與stdout一樣, 發送到一個地方.
64
65 i>&j
66 # 重定向文件描述符i 到 j.
67 # 指向i文件的所有輸出都發送到j中去.
68
69 >&j
70 # 默認的, 重定向文件描述符1(stdout)到 j.
71 # 所有傳遞到stdout的輸出都送到j中去.
72
73 0< FILENAME
74 < FILENAME
75 # 從文件中接受輸入.
76 # 與">"是成對命令, 並且通常都是結合使用.
77 #
78 # grep search-word <filename
79
80
81 [j]<>filename
82 # 為了讀寫"filename", 把文件"filename"打開, 並且分配文件描述符"j"給它.
83 # 如果文件"filename"不存在, 那麼就創建它.
84 # 如果文件描述符"j"沒指定, 那默認是fd 0, stdin.
85 #
86 # 這種應用通常是為了寫到一個文件中指定的地方.
87 echo 1234567890 > File # 寫字符串到"File".
88 exec 3<> File # 打開"File"並且給它分配fd 3.
89 read -n 4 <&3 # 只讀4個字符.
90 echo -n . >&3 # 寫一個小數點.
91 exec 3>&- # 關閉fd 3.
92 cat File # ==> 1234.67890
93 # 隨機存儲.
94
95
96
97 |
98 # 管道.
99 # 通用目的的處理和命令鏈工具.
100 # 與">"很相似, 但是實際上更通用.
101 # 對於想將命令, 腳本, 文件和程序串連起來的時候很有用.
102 cat *.txt | sort | uniq > result-file
103 # 對所有的.txt文件的輸出進行排序, 並且刪除重復行,
104 # 最後將結果保存到"result-file"中.