Shell 命令執(zhí)行順序分析[圖]
更新時間:2013年01月23日 16:15:36 作者:
Shell 從標(biāo)準(zhǔn)輸入或腳本中讀取的每一行稱為管道(pipeline);它包含了一個或多個命令(command),這些命令被一個或多個管道字符(|)隔開
Shell 從標(biāo)準(zhǔn)輸入或腳本中讀取的每一行稱為管道(pipeline);它包含了一個或多個命令(command),這些命令被一個或多個管道字符(|)隔開。
事實上還有很多特殊符號可用來分隔單個的命令:分號(;)、管道(|)、&、邏輯AND (&&),還有邏輯OR (||)。對于每一個讀取的管道,Shell都回將命令分割,為管道設(shè)置I/O,并且對每一個命令依次執(zhí)行下面的操作:

整個步驟順序如上圖所示,看起來有些復(fù)雜。當(dāng)命令行被處理時,每一個步驟都是在Shell的內(nèi)存里發(fā)生的;Shell不會真的把每個步驟的發(fā)生顯示給你看。所以,你可以假想這事我們偷窺Shell內(nèi)存里的情況,從而知道每個階段的命令行是如何被轉(zhuǎn)換的。我們從這個例子開始說:
$ mkidr /tmp/x 建立臨時性目錄
$ cd /tmp/x 切換到該目錄
$ touch f1 f2 建立文件
$ f=f y="a b" 賦值兩個變量
$ echo ~+/${f}[12] $y $(echo cmd subst )$ (( 3 + 2 )) > out 將結(jié)果重定向到out
上述的執(zhí)行步驟概要如下:
1.命令一開始回根據(jù)Shell語法而分割為token。最重要的一點是:I/O重定向 >out 在這里是被識別的,并存儲供稍后使用。流程繼續(xù)處理下面這行,其中每個token的范圍顯示于命令下面的行上:
echo ~+/${f}[12] $y $(echo cmd subst) $((3 + 2))
| 1 | |----- 2 ----| |3 | |-------- 4----------| |----5-----|
2.檢查第一個單詞(echo)是否為關(guān)鍵字,例如 if 或 for 。這里不是,所以命令行不變繼續(xù)處理。
3.檢查第一個單詞(echo)是否為別名。這里不是。所以命令行不變,繼續(xù)處理。
4.掃描所以單詞是否需要波浪號展開。在本例中,~+ 為ksh93 與 bash 的擴(kuò)展,等同于$PWD,也就是當(dāng)前的目錄。token 2將被修改,處理如下:
echo /tmp/x/${f}[12] $y $(echo cmd subst) $((3 + 2))
| 1 | |------- 2 -------| |3 | |-------- 4----------| |----5-----|
5.下一步是變量展開:token 2 與 3 都被修改。這樣會產(chǎn)生:
echo /tmp/x/${f}[12] a b $(echo cmd subst) $((3 + 2))
| 1 | |------- 2 -------| | 3 | |-------- 4----------| |----5-----|
6.再來要處理的是命令替換。注意,這里可用遞歸應(yīng)用列表里的所有步驟!在這里,命令替換修改了 token 4:
echo /tmp/x/${f}[12] a b cmd subst $((3 + 2))
| 1 | |------- 2 -------| | 3 | |--- 4 ----| |----5-----|
7.現(xiàn)在執(zhí)行算數(shù)替換。修改的是 token 5,結(jié)果:
echo /tmp/x/${f}[12] a b cmd subst 5
| 1 | |------- 2 -------| | 3 | |--- 4 ----| |5|
8.前面所有的展開產(chǎn)生的結(jié)果,都將再一次被掃描,看看是否有 $IFS 字符。如果有,則他們是作為分隔符(separator),產(chǎn)生額外的單詞,例如,兩個字符$y 原來是組成一個單詞,單展開式“a- 空格-b”,在此階段被切分為兩個單詞:a 與 b。相同方式也應(yīng)用于命令$(echo cmd subst)的結(jié)果上。先前的 token 3 變成了 token 3 與
token 4.先前的 token 4則成了 token 5 與 token 6。結(jié)果:
echo /tmp/x/${f}[12] a b cmd subst 5
| 1 | |------- 2 -------| 3 4 |-5-| |- 6 -| 7
9.最后的替換階段是通配符展開。token 2 變成了 token 2 與 token 3:
echo /tmp/x/$f1 /tmp/x/$f2 a b cmd subst 5
| 1 | |---- 2 ----| |---- 3 ----| 4 5 |-6-| |- 7 -| 8
10.這時,Shell已經(jīng)準(zhǔn)備好了要執(zhí)行最后的命令了。它會去尋找 echo。正好 ksh93 與 bash 的 echo 都內(nèi)建到Shell 中了。
11.Shell實際執(zhí)行命令。首先執(zhí)行 > out 的 I/O重定向,再調(diào)用內(nèi)部的 echo 版本,顯示最后的參數(shù)。
最后的結(jié)果:
$cat out
/tmp/x/f1 /tmp/x/f2 a b cmd subst 5
事實上還有很多特殊符號可用來分隔單個的命令:分號(;)、管道(|)、&、邏輯AND (&&),還有邏輯OR (||)。對于每一個讀取的管道,Shell都回將命令分割,為管道設(shè)置I/O,并且對每一個命令依次執(zhí)行下面的操作:

整個步驟順序如上圖所示,看起來有些復(fù)雜。當(dāng)命令行被處理時,每一個步驟都是在Shell的內(nèi)存里發(fā)生的;Shell不會真的把每個步驟的發(fā)生顯示給你看。所以,你可以假想這事我們偷窺Shell內(nèi)存里的情況,從而知道每個階段的命令行是如何被轉(zhuǎn)換的。我們從這個例子開始說:
復(fù)制代碼 代碼如下:
$ mkidr /tmp/x 建立臨時性目錄
$ cd /tmp/x 切換到該目錄
$ touch f1 f2 建立文件
$ f=f y="a b" 賦值兩個變量
$ echo ~+/${f}[12] $y $(echo cmd subst )$ (( 3 + 2 )) > out 將結(jié)果重定向到out
上述的執(zhí)行步驟概要如下:
1.命令一開始回根據(jù)Shell語法而分割為token。最重要的一點是:I/O重定向 >out 在這里是被識別的,并存儲供稍后使用。流程繼續(xù)處理下面這行,其中每個token的范圍顯示于命令下面的行上:
echo ~+/${f}[12] $y $(echo cmd subst) $((3 + 2))
| 1 | |----- 2 ----| |3 | |-------- 4----------| |----5-----|
2.檢查第一個單詞(echo)是否為關(guān)鍵字,例如 if 或 for 。這里不是,所以命令行不變繼續(xù)處理。
3.檢查第一個單詞(echo)是否為別名。這里不是。所以命令行不變,繼續(xù)處理。
4.掃描所以單詞是否需要波浪號展開。在本例中,~+ 為ksh93 與 bash 的擴(kuò)展,等同于$PWD,也就是當(dāng)前的目錄。token 2將被修改,處理如下:
echo /tmp/x/${f}[12] $y $(echo cmd subst) $((3 + 2))
| 1 | |------- 2 -------| |3 | |-------- 4----------| |----5-----|
5.下一步是變量展開:token 2 與 3 都被修改。這樣會產(chǎn)生:
echo /tmp/x/${f}[12] a b $(echo cmd subst) $((3 + 2))
| 1 | |------- 2 -------| | 3 | |-------- 4----------| |----5-----|
6.再來要處理的是命令替換。注意,這里可用遞歸應(yīng)用列表里的所有步驟!在這里,命令替換修改了 token 4:
echo /tmp/x/${f}[12] a b cmd subst $((3 + 2))
| 1 | |------- 2 -------| | 3 | |--- 4 ----| |----5-----|
7.現(xiàn)在執(zhí)行算數(shù)替換。修改的是 token 5,結(jié)果:
echo /tmp/x/${f}[12] a b cmd subst 5
| 1 | |------- 2 -------| | 3 | |--- 4 ----| |5|
8.前面所有的展開產(chǎn)生的結(jié)果,都將再一次被掃描,看看是否有 $IFS 字符。如果有,則他們是作為分隔符(separator),產(chǎn)生額外的單詞,例如,兩個字符$y 原來是組成一個單詞,單展開式“a- 空格-b”,在此階段被切分為兩個單詞:a 與 b。相同方式也應(yīng)用于命令$(echo cmd subst)的結(jié)果上。先前的 token 3 變成了 token 3 與
token 4.先前的 token 4則成了 token 5 與 token 6。結(jié)果:
echo /tmp/x/${f}[12] a b cmd subst 5
| 1 | |------- 2 -------| 3 4 |-5-| |- 6 -| 7
9.最后的替換階段是通配符展開。token 2 變成了 token 2 與 token 3:
echo /tmp/x/$f1 /tmp/x/$f2 a b cmd subst 5
| 1 | |---- 2 ----| |---- 3 ----| 4 5 |-6-| |- 7 -| 8
10.這時,Shell已經(jīng)準(zhǔn)備好了要執(zhí)行最后的命令了。它會去尋找 echo。正好 ksh93 與 bash 的 echo 都內(nèi)建到Shell 中了。
11.Shell實際執(zhí)行命令。首先執(zhí)行 > out 的 I/O重定向,再調(diào)用內(nèi)部的 echo 版本,顯示最后的參數(shù)。
最后的結(jié)果:
復(fù)制代碼 代碼如下:
$cat out
/tmp/x/f1 /tmp/x/f2 a b cmd subst 5
相關(guān)文章
linux環(huán)境下編寫shell腳本實現(xiàn)啟動停止tomcat服務(wù)的方法
這篇文章主要介紹了linux環(huán)境下編寫shell腳本實現(xiàn)啟動停止tomcat服務(wù)的方法,非常不錯,具有一定的參考借鑒價值,需要的朋友可以參考下2018-06-06Shell創(chuàng)建用戶并生成隨機(jī)密碼腳本分享
這篇文章主要介紹了Shell創(chuàng)建用戶并生成隨機(jī)密碼腳本分享,本文生成的隨機(jī)密碼會比較復(fù)雜和實用,需要的朋友可以參考下2014-12-12Shell腳本中判斷輸入?yún)?shù)個數(shù)的方法
這篇文章主要介紹了Shell腳本中判斷輸入?yún)?shù)個數(shù)的方法,使用內(nèi)置變量$#即可實現(xiàn)判斷輸入了多少個參數(shù),需要的朋友可以參考下2014-10-10