本文也即《Learning the bash Shell》3rd Edition的第五章Flow Control之讀書筆記,但我們將不限於此。flow control是任何編程語言中很常用的部分,也包括了bash。在這里,我們將學習他們。
if/else是通過判斷選擇執行或者執行部分代碼,可以根據變量、文件名、命令是否執行成功等很多條件進行判斷,他的格式如下:
if condition
then
statements
[elif condition
then statements. ..]
[else
statements ]
fi
和 C程序不一樣,bash的判斷不是通過boolean,而是通過statement,也就是執行命令后的最終狀態(exit status)。所有的Linux命令,無論你是代碼是C還是腳本,執行完,都返回一個整數通知他的調用這,這就是exit status,通常0表示OK,其他(1-255)表示錯誤。這只是通常的情況,例如diff,0表示你no difference,1表示difference,2表示錯誤。if判斷statements的最后一個的exit status,通常我們只放一個statement,如果為0,表示true,否則表示false。
執行下一條命令會沖掉原來exit status。可以使用$?來查看上一命令執行的結果。例如我們希望用一個新的cd命令來替代原來在linux kernel中已將編譯的cd命令,由於function是優先於built-in命令,所以調用時,將調用我們的function。下面有一個例 子,function pushd,在stack中鍵入cd的dirname路徑名,並執行跳到該路徑下。
cd ( )
{
#由於我們已經定義了具有更高優先級別的function,如果希望調用原來built-in的命令,需要再前面加上builtin。
builtin cd "$@"
#$?是上一command的返回值,即builtin cd "$@"的值,並記錄在result里面。
result=$?
echo "$OLDPWD --> $PWD"
#返回result的值。我們需要注意shell中的返回和在其他程序,例如C語 言中的返回是不一樣的,只代表最后的exit statue,而不是所謂的返回值,雖然也用到了return。如何沒有最后的reture,例如后面的push_func,exit status就是最后執行的command的exit status
return $result
}push_func( )
{
dirname=$1
#如果dirname為null,退出funcuntion,如cd dirname成功,push the directory ,否則顯示still in $PWD,cd使用function的cd函數,其優先級別高於已在內核編譯了的cd
if cd ${dirname:?"missing directory name."}
then
mystack="$dirname ${mystack:-$OLDPWD }"
echo $mystack
else
echo still in $PWD.
fi
}push_func $1
條件結合
和C語言一樣,可以進行條件結合,使用&&,||,以及!三種方式,表示“和”,“或”,與”非“,格式如下:if statement1 && statement2, if statement1 || statement2 ,if ! statement1 。
exit status不是判斷的唯一值,可以使用[...]和[[...]]。
字符串比較
字符串比較是放置在[...]中,有以下的幾種:
- str1 = str2,字符串1匹配字符串2
- str1 != str2,字符串1不匹配字符串2
- str1 > str2,字符串1大於字符串2
- str1 < str2,字符串1小於字符串2
- -n str,字符串不為null,長度大於零
- -z str,字符串為null,長度為零
需要注意<和>符號和重定向符號相似,為了避免歧義和錯誤,使用if [ $a /> $b ] 的方式 。仍然上面的例子,我們增加pop_func來操作stack:
pop_func()
{
mystack=${mystack#* }
#下面請至於[ ... ],即[后面有一個空格,]前面有一空格,另外$mystack用雙引號,表示這代表的是一個字符串。注意到then不放在下一行,與if放在用一行,用;來隔開。
if [ -n "$mystack" ]; then
cd ${mystack%% *}
echo "$PWD", stack is [$mystack]
else
echo "stack empty, still in $PWD."
fi
}
例如,我們要求命令帶有參數,除了使用{1?"<message"}以外,下面給出更可讀的方式:
if [ -z "$1" ]; then
echo 'usage: c filename [-N]'
exit 1
fi
在這里exit表示結束,退出,執行的結果為失敗,非零。
文件屬性比較
文件屬性比較是另一個常用的條件判斷類型。
- -a file :file 存在
- -d file :file存在並是一個目錄
- -e file :file 存在,同- a
- -f file :file 存在並且是一個常規的文件(不是目錄或者其他特殊類型文件)
- -r file :有讀的權限
- -s file :文件存在且不為空
- -w file :有寫的權限
- -x file :有執行的權限,或者對於目錄有search的權限
- -N file :在上次讀取后,文件有改動
- -O file :own所屬的文件
- -G file :group所屬的文件
- file1 -nt file2 :file1 比 file2 更新,以最后更新時間為准
- file1 -ot file2 :file1 比 file2 更舊 ,以最后更新時間為准
這些在[ ... ]中的條件判斷是可以多個結合起來,例如if [ condition ] && [ condition ]; then,當然也可以if command && [ condition ]; then,不在類推。尤其我們可以進行復制的條件判斷。另外還可以使用-a 和-o ,等同於C語言中的&和|的邏輯計算復符號,他們和&&即||相似。當他們用在condition里面。
在上面push_func的例子中,除了判斷是否參數之外,增加判斷是否是目錄名,如下:
if [ -n "$dirname" ] &&[ -d "$dirname" ]
then
cd $dirname
mystack="$dirname ${mystack:-$OLDPWD }"
echo $mystack
else
echo still in $PWD.
fi
我們在增加一個判斷,當時目錄名的時候,在檢查是否可以進行查看或操作。使用if [ -n "$dirname" ] &&[ -d "$dirname" -a -x "$dirname" ],但是這種寫法很難閱讀,我們需要將兩個前后判斷括起來,( -d "$dirname" ) -a ( -x "$dirname" )。但是(是個特殊符合,需使用/(的方式,即為:if [ -n "$dirname" ] &&[ /( -d "$dirname" /) -a /( -x "$dirname" /) ] 。
整數比較
>或者<或者=是用於字符串的比較,如果用於整數比較,使用:
- -lt,小於
- -le,小於等於
- -eq,等於
- -ge,大於等於
- -gt,大於
- -ne,不等於
