關(guān)于linux bash的重定向
廣東網(wǎng)站建設(shè)公司創(chuàng)新互聯(lián),廣東網(wǎng)站設(shè)計(jì)制作,有大型網(wǎng)站制作公司豐富經(jīng)驗(yàn)。已為廣東數(shù)千家提供企業(yè)網(wǎng)站建設(shè)服務(wù)。企業(yè)網(wǎng)站搭建\外貿(mào)營(yíng)銷網(wǎng)站建設(shè)要多少錢(qián),請(qǐng)找那個(gè)售后服務(wù)好的廣東做網(wǎng)站的公司定做!
所謂I/O重定向簡(jiǎn)單來(lái)說(shuō)就是一個(gè)過(guò)程,這個(gè)過(guò)程捕捉一個(gè)文件,或者命令,程序,腳本,甚至腳本中的代碼塊(code block)的輸出,然后把捕捉到的輸出,作為輸入發(fā)送給另外一個(gè)文件,命令,程序,或者腳本。
如果談到I/O重定向,就涉及到文件標(biāo)識(shí)符(File Descriptor)的概念, 在Linux系統(tǒng)中,系統(tǒng)為每一個(gè)打開(kāi)的文件指定一個(gè)文件標(biāo)識(shí)符以便系統(tǒng)對(duì)文件進(jìn)行跟蹤,這里有些和C語(yǔ)言編程里的文件句柄相似,文件標(biāo)識(shí)符是一個(gè)數(shù)字,不同數(shù)字代表不同的含義,默認(rèn)情況下,系統(tǒng)占用了3個(gè),分別是0標(biāo)準(zhǔn)輸入(stdin),1標(biāo)準(zhǔn)輸出(stdout), 2標(biāo)準(zhǔn)錯(cuò)誤(stderr), 另外3-9是保留的標(biāo)識(shí)符,可以把這些標(biāo)識(shí)符指定成標(biāo)準(zhǔn)輸入,輸出或者錯(cuò)誤作為臨時(shí)連接。通常這樣可以解決很多復(fù)雜的重定向請(qǐng)求。
標(biāo)準(zhǔn)輸入通常指鍵盤(pán)的輸入
標(biāo)準(zhǔn)輸出通常指顯示器的輸出
標(biāo)準(zhǔn)錯(cuò)誤通常也是定向到顯示器
請(qǐng)看以下例子,來(lái)描述一下他們的關(guān)系
#ls /dev
這個(gè)命令列出/dev目錄下的所有文件,在屏幕上輸出結(jié)果。
這里 /dev 就是作為命令ls的標(biāo)準(zhǔn)輸入(從鍵盤(pán)輸入),而打印在屏幕的結(jié)果就是標(biāo)準(zhǔn)輸出(/dev目錄中的內(nèi)容)
還是回到標(biāo)題,重定向就是把標(biāo)準(zhǔn)的`輸入或者輸出更改成其他的方式,請(qǐng)參看如下例子
或者等同于
#ls /dev 1filename #注意:"1"和""中間沒(méi)有空格
以上命令會(huì)把命令的標(biāo)準(zhǔn)輸出重新定向到一個(gè)文件filename,而不是顯示到屏幕上,如果不指明文件標(biāo)識(shí)符,系統(tǒng)默認(rèn)的就是1, 因此1可以省略
如果把上面例子重的""改成""則表示把輸出追加到filename文件的末尾,如果文件不存在則創(chuàng)建它。如下
#ls /dev filename
也可以把標(biāo)準(zhǔn)錯(cuò)誤重新定向到文件
#ls -qw /dev 2filename
顯然 -qw是一個(gè)錯(cuò)誤參數(shù),通常會(huì)在顯示器上報(bào)告一個(gè)錯(cuò)誤信息,但由于把2標(biāo)準(zhǔn)錯(cuò)誤(stderr)重新定向到了文件filename,因此顯示器沒(méi)有錯(cuò)誤信息,而信息寫(xiě)到了文件里面
以下命令是把標(biāo)準(zhǔn)輸出和錯(cuò)誤都定向到文件
#ls /dev filename
""在這里代表標(biāo)準(zhǔn)輸出和標(biāo)準(zhǔn)錯(cuò)誤,這里無(wú)論是正常輸出還是錯(cuò)誤信息都寫(xiě)到filename中了。
重新定義標(biāo)準(zhǔn)輸入,輸出,和錯(cuò)誤的文件標(biāo)識(shí)符
重新定義文件標(biāo)識(shí)符可以用ij命令,表示把文件標(biāo)識(shí)符i重新定向到j(luò),你可以把""理解為"取地址"
請(qǐng)看以下例子
#exec 51
表示把文件標(biāo)識(shí)符5定向到標(biāo)準(zhǔn)輸出,這個(gè)命令通常用來(lái)臨時(shí)保存標(biāo)準(zhǔn)輸入。
同樣標(biāo)準(zhǔn)輸入也是可以重新定向的,請(qǐng)參考下面例子
# grep search-word
一般來(lái)說(shuō)grep命令在給定文件中搜索字符串,以上命令把文件filename作為grep命令的標(biāo)準(zhǔn)輸入,而不是從鍵盤(pán)輸入。
前面曾經(jīng)提到,系統(tǒng)為每一個(gè)打開(kāi)的文件指定一個(gè)文件標(biāo)識(shí)符以便系統(tǒng)對(duì)文件進(jìn)行跟蹤,那么默認(rèn)的文件標(biāo)識(shí)符是什么呢?答案是0,也就是標(biāo)準(zhǔn)輸入,或者可以說(shuō)從鍵盤(pán)輸入。當(dāng)然這個(gè)文件標(biāo)識(shí)符也可以自己指定,請(qǐng)參考下面例子
#echo 123456789 filename 把字符串寫(xiě)到文件filename中 #exec 3filename 把文件filename打開(kāi),并指定文件標(biāo)識(shí)符為3 #read -n 4 3 從文件中讀4個(gè)字符,句柄已經(jīng)指到第四個(gè)字符末尾 #echo -n . 3 在第5個(gè)字符處寫(xiě)一個(gè)點(diǎn),覆蓋第5個(gè)字符,-n表示不換行 #exec 3- 關(guān)閉文件標(biāo)識(shí)符3
現(xiàn)在cat filename文件的結(jié)果就成了1234.6789
命令jfilename表示把文件打開(kāi),并指明文件標(biāo)識(shí)符為j
"-"表示關(guān)閉文件標(biāo)識(shí)符
有關(guān)關(guān)閉文件標(biāo)識(shí)符的操作請(qǐng)參考下面
n- 關(guān)閉輸入文件標(biāo)識(shí)符n 0-或- 關(guān)閉標(biāo)準(zhǔn)輸入stdin n- 關(guān)閉輸出文件標(biāo)識(shí)符n 1-或-關(guān)閉標(biāo)準(zhǔn)輸出stdout
另外還有一些其他命令,如下參考
2. : filename 或者 filename
表示把文件filename設(shè)置成空,也就是清空文件內(nèi)容,如果文件不存在,則創(chuàng)建一個(gè)空文件,(等同于touch命令) :表示一個(gè)空輸出,兩個(gè)命令的唯一區(qū)別就是filename不是在所有shell都可以正常工作的。
;
一個(gè)進(jìn)程默認(rèn)會(huì)打開(kāi)標(biāo)準(zhǔn)輸入、標(biāo)準(zhǔn)輸出、錯(cuò)誤輸出三個(gè)文件描述符
文件描述符為0,小于號(hào)表示數(shù)據(jù)由文件流入命令
usage:
命令 file
#加入輸入重定向,對(duì)輸入文件進(jìn)行計(jì)數(shù)
wc -l /etc/passwd
文件描述符為1,大于號(hào)表述數(shù)據(jù)由命令流入文件
所以: echo "hello" t.log 也可以寫(xiě)成 echo "hello" 1 t.log
會(huì)把文件的內(nèi)容清空,再輸入到新的文件
會(huì)把當(dāng)前內(nèi)容保持不變,然后在下一行進(jìn)行追加
2 錯(cuò)誤重定向,如果當(dāng)前命令產(chǎn)生了錯(cuò)誤,會(huì)把錯(cuò)誤重定向到文件中,而不會(huì)打印到屏幕
無(wú)論內(nèi)容正確還是錯(cuò)誤都會(huì)把文件輸出到文件中。
stderr的文件描述符為2,Unix程序會(huì)向stderr流中寫(xiě)入錯(cuò)誤信息。
默認(rèn)情況下, command file 將stdout 重定向到 file, command file 將stdin 重定向到 file。
如果希望 stderr 重定向到 file,可以這樣寫(xiě):
$ command 2file
如果希望 stderr 追加到 file 文件末尾,可以這樣寫(xiě):
$ command 2file 2 表示標(biāo)準(zhǔn)錯(cuò)誤文件(stderr)
如果希望將 stdout 和 stderr 合并后重定向到 file,可以這樣寫(xiě):
$ command file 21 或者 $ command file 21
如果希望對(duì) stdin 和 stdout 都重定向,可以這樣寫(xiě):
$ command file1 file2
command 命令將 stdin 重定向到 file1,將 stdout 重定向到 file2。
含義:將標(biāo)準(zhǔn)錯(cuò)誤輸出 重定向到 標(biāo)準(zhǔn)輸出。
符號(hào)是一個(gè)整體,不可分開(kāi),分開(kāi)后就不是上述含義了。
比如有些人可能會(huì)這么想:2是標(biāo)準(zhǔn)錯(cuò)誤輸入,1是標(biāo)準(zhǔn)輸出,是重定向符號(hào),那么"將標(biāo)準(zhǔn)錯(cuò)誤輸出重定向到標(biāo)準(zhǔn)輸出"是不是就應(yīng)該寫(xiě)成"21"就行了?是這樣嗎? 如果是嘗試過(guò),你就知道21的寫(xiě)法其實(shí)是將標(biāo)準(zhǔn)錯(cuò)誤輸出重定向到名為"1"的文件里去了
用途:不掛斷地運(yùn)行命令。
語(yǔ)法:nohup Command [ Arg … ] [ ]
無(wú)論是否將 nohup 命令的輸出重定向到終端,輸出都將附加到當(dāng)前目錄的 nohup.out 文件中。
如果當(dāng)前目錄的 nohup.out 文件不可寫(xiě),輸出重定向到 $HOME/nohup.out 文件中。
如果沒(méi)有文件能創(chuàng)建或打開(kāi)以用于追加,那么 Command 參數(shù)指定的命令不可調(diào)用。
退出狀態(tài):該命令返回下列出口值:
126 可以查找但不能調(diào)用 Command 參數(shù)指定的命令。
127 nohup 命令發(fā)生錯(cuò)誤或不能查找由 Command 參數(shù)指定的命令。
否則,nohup 命令的退出狀態(tài)是 Command 參數(shù)指定命令的退出狀態(tài)。
用途:在后臺(tái)運(yùn)行
一般兩個(gè)一起用
3.為什么呢 21 要寫(xiě)在后面?
首先是command file將標(biāo)準(zhǔn)輸出重定向到file中, 21 是標(biāo)準(zhǔn)錯(cuò)誤拷貝了標(biāo)準(zhǔn)輸出的行為,也就是同樣被重定向到file中,最終結(jié)果就是標(biāo)準(zhǔn)輸出和錯(cuò)誤都被重定向到file中。
21 標(biāo)準(zhǔn)錯(cuò)誤拷貝了標(biāo)準(zhǔn)輸出的行為,但此時(shí)標(biāo)準(zhǔn)輸出還是在終端。file 后輸出才被重定向到file,但標(biāo)準(zhǔn)錯(cuò)誤仍然保持在終端。
用strace可以看到:
這個(gè)命令中實(shí)現(xiàn)重定向的關(guān)鍵系統(tǒng)調(diào)用序列是:
open(file) == 3
dup2(3,1)
dup2(1,2)
這個(gè)命令中實(shí)現(xiàn)重定向的關(guān)鍵系統(tǒng)調(diào)用序列是:
dup2(1,2)
open(file) == 3
dup2(3,1)
注意:0 是標(biāo)準(zhǔn)輸入(STDIN),1 是標(biāo)準(zhǔn)輸出(STDOUT),2 是標(biāo)準(zhǔn)錯(cuò)誤輸出(STDERR)。
這里的2和之間不可以有空格,2是一體的時(shí)候才表示錯(cuò)誤輸出。