[shell] if else以及大於、小於、等於邏輯表達式 [轉]


本文也即《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,不等於

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM