操作系統知識(8)--Shell(3)

Shell test命令

Shell中的 test 命令用於檢查某個條件是否成立,它可以進行數值、字符和文件三個方面的測試。

數值測試

這裏寫圖片描述

實例演示:

num1=100
num2=100
if test $[num1] -eq $[num2]
then
    echo '兩個數相等!'
else
    echo '兩個數不相等!'
fi

輸出結果:

兩個數相等!

字符串測試

這裏寫圖片描述

實例演示:

num1="runoob"
num2="runoob"
if test num1=num2
then
    echo '兩個字符串相等!'
else
    echo '兩個字符串不相等!'
fi

輸出結果:

兩個字符串相等!

文件測試

這裏寫圖片描述

實例演示:

cd /bin
if test -e ./bash
then
    echo '文件已存在!'
else
    echo '文件不存在!'
fi

輸出結果:

文件已存在!

另外,Shell還提供了與( -a )、或( -o )、非( ! )三個邏輯操作符用於將測試條件連接起來,其優先級爲:”!”最高,”-a”次之,”-o”最低。例如:

cd /bin
if test -e ./notFile -o -e ./bash
then
    echo '有一個文件存在!'
else
    echo '兩個文件都不存在'
fi

輸出結果:

有一個文件存在!

Shell 流程控制

和Java、PHP等語言不一樣,sh的流程控制不可爲空,如(以下爲PHP流程控制寫法):

<?php
if (isset($_GET["q"])) {
    search(q);
}
else {
    // 不做任何事情
}

在sh/bash裏可不能這麼寫,如果else分支沒有語句執行,就不要寫這個else。

if else

if

if 語句語法格式:

if condition
then
    command1 
    command2
    ...
    commandN 
fi

寫成一行(適用於終端命令提示符):

if [ $(ps -ef | grep -c "ssh") -gt 1 ]; then echo "true"; fi

末尾的fi就是if倒過來拼寫,後面還會遇到類似的。

if else

if else 語法格式:

if condition
then
    command1 
    command2
    ...
    commandN
else
    command
fi

if else-if else

if else-if else 語法格式:

if condition1
then
    command1
elif condition2 
then 
    command2
else
    commandN
fi

以下實例判斷兩個變量是否相等:

a=10
b=20
if [ $a == $b ]
then
   echo "a 等於 b"
elif [ $a -gt $b ]
then
   echo "a 大於 b"
elif [ $a -lt $b ]
then
   echo "a 小於 b"
else
   echo "沒有符合的條件"
fi

輸出結果:

a 小於 b

if else語句經常與test命令結合使用,如下所示:

num1=$[2*3]
num2=$[1+5]
if test $[num1] -eq $[num2]
then
    echo '兩個數字相等!'
else
    echo '兩個數字不相等!'
fi

輸出結果:

兩個數字相等!

for 循環

與其他編程語言類似,Shell支持for循環。

for循環一般格式爲:

for var in item1 item2 ... itemN
do
    command1
    command2
    ...
    commandN
done

寫成一行:

for var in item1 item2 ... itemN; do command1; command2… done;

當變量值在列表裏,for循環即執行一次所有命令,使用變量名獲取列表中的當前取值。命令可爲任何有效的shell命令和語句。in列表可以包含替換、字符串和文件名。

in列表是可選的,如果不用它,for循環使用命令行的位置參數。

例如,順序輸出當前列表中的數字:

for loop in 1 2 3 4 5
do
    echo "The value is: $loop"
done

輸出結果:

The value is: 1
The value is: 2
The value is: 3
The value is: 4
The value is: 5

順序輸出字符串中的字符:

for str in 'This is a string'
do
    echo $str
done

輸出結果:

This is a string

while 語句

while循環用於不斷執行一系列命令,也用於從輸入文件中讀取數據;命令通常爲測試條件。其格式爲:

while condition
do
    command
done

以下是一個基本的while循環,測試條件是:如果int小於等於5,那麼條件返回真。int從0開始,每次循環處理時,int加1。運行上述腳本,返回數字1到5,然後終止。

#!/bin/sh
int=1
while(( $int<=5 ))
do
        echo $int
        let "int++"
done

運行腳本,輸出:

1
2
3
4
5

使用中使用了 Bash let 命令,它用於執行一個或多個表達式,變量計算中不需要加上 $ 來表示變量,具體可查閱:Bash let 命令

while循環可用於讀取鍵盤信息。下面的例子中,輸入信息被設置爲變量FILM,按結束循環。

echo '按下 <CTRL-D> 退出'
echo -n '輸入你最喜歡的電影名: '
while read FILM
do
    echo "是的!$FILM 是一部好電影"
done

運行腳本,輸出類似下面:

按下 <CTRL-D> 退出
輸入你最喜歡的電影名: w3cschool菜鳥教程
是的!w3cschool菜鳥教程 是一部好電影

無限循環

無限循環語法格式:

while :
do
    command
done

或者

while true
do
    command
done

或者

for (( ; ; ))

until 循環

until循環執行一系列命令直至條件爲真時停止。

until循環與while循環在處理方式上剛好相反。

一般while循環優於until循環,但在某些時候—也只是極少數情況下,until循環更加有用。

until 語法格式:

until condition
do
    command
done

條件可爲任意測試條件,測試發生在循環末尾,因此循環至少執行一次—請注意這一點。

case

Shell case語句爲多選擇語句。可以用case語句匹配一個值與一個模式,如果匹配成功,執行相匹配的命令。case語句格式如下:

case 值 in
模式1)
    command1
    command2
    ...
    commandN
    ;;
模式2)
    command1
    command2
    ...
    commandN
    ;;
esac

case工作方式如上所示。取值後面必須爲單詞in,每一模式必須以右括號結束。取值可以爲變量或常數。匹配發現取值符合某一模式後,其間所有命令開始執行直至 ;;。

取值將檢測匹配的每一個模式。一旦模式匹配,則執行完匹配模式相應命令後不再繼續其他模式。如果無一匹配模式,使用星號 * 捕獲該值,再執行後面的命令。

下面的腳本提示輸入1到4,與每一種模式進行匹配:

echo '輸入 1 到 4 之間的數字:'
echo '你輸入的數字爲:'
read aNum
case $aNum in
    1)  echo '你選擇了 1'
    ;;
    2)  echo '你選擇了 2'
    ;;
    3)  echo '你選擇了 3'
    ;;
    4)  echo '你選擇了 4'
    ;;
    *)  echo '你沒有輸入 1 到 4 之間的數字'
    ;;
esac

輸入不同的內容,會有不同的結果,例如:

輸入 1 到 4 之間的數字:
你輸入的數字爲:
3
你選擇了 3

跳出循環

在循環過程中,有時候需要在未達到循環結束條件時強制跳出循環,Shell使用兩個命令來實現該功能:break和continue。

break命令

break命令允許跳出所有循環(終止執行後面的所有循環)。

下面的例子中,腳本進入死循環直至用戶輸入數字大於5。要跳出這個循環,返回到shell提示符下,需要使用break命令。

#!/bin/bash
while :
do
    echo -n "輸入 1 到 5 之間的數字:"
    read aNum
    case $aNum in
        1|2|3|4|5) echo "你輸入的數字爲 $aNum!"
        ;;
        *) echo "你輸入的數字不是 1 到 5 之間的! 遊戲結束"
            break
        ;;
    esac
done

執行以上代碼,輸出結果爲:

輸入 1 到 5 之間的數字:3
你輸入的數字爲 3!
輸入 1 到 5 之間的數字:7
你輸入的數字不是 1 到 5 之間的! 遊戲結束

continue

continue命令與break命令類似,只有一點差別,它不會跳出所有循環,僅僅跳出當前循環。
對上面的例子進行修改:

#!/bin/bash
while :
do
    echo -n "輸入 1 到 5 之間的數字: "
    read aNum
    case $aNum in
        1|2|3|4|5) echo "你輸入的數字爲 $aNum!"
        ;;
        *) echo "你輸入的數字不是 1 到 5 之間的!"
            continue
            echo "遊戲結束"
        ;;
    esac
done

運行代碼發現,當輸入大於5的數字時,該例中的循環不會結束,語句 echo “Game is over!” 永遠不會被執行。

esac

case的語法和C family語言差別很大,它需要一個esac(就是case反過來)作爲結束標記,每個case分支用右圓括號,用兩個分號表示break。

Shell 函數

linux shell 可以用戶定義函數,然後在shell腳本中可以隨便調用。
shell中函數的定義格式如下:

[ function ] funname [()]

{

    action;

    [return int;]

}

說明:

  • 1、可以帶function fun() 定義,也可以直接fun() 定義,不帶任何參數。
  • 2、參數返回,可以顯示加:return 返回,如果不加,將以最後一條命令運行結果,作爲返回值。 return後跟數值n(0-255
  • 下面的例子定義了一個函數並進行調用:

    #!/bin/bash
    
    demoFun(){
        echo "這是我的第一個 shell 函數!"
    }
    echo "-----函數開始執行-----"
    demoFun
    echo "-----函數執行完畢-----"

    輸出結果:

    -----函數開始執行-----
    這是我的第一個 shell 函數!
    -----函數執行完畢-----

    下面定義一個帶有return語句的函數:

    #!/bin/bash
    
    funWithReturn(){
        echo "這個函數會對輸入的兩個數字進行相加運算..."
        echo "輸入第一個數字: "
        read aNum
        echo "輸入第二個數字: "
        read anotherNum
        echo "兩個數字分別爲 $aNum$anotherNum !"
        return $(($aNum+$anotherNum))
    }
    funWithReturn
    echo "輸入的兩個數字之和爲 $? !"

    輸出類似下面:

    這個函數會對輸入的兩個數字進行相加運算...
    輸入第一個數字: 
    1
    輸入第二個數字: 
    2
    兩個數字分別爲 12 !
    輸入的兩個數字之和爲 3 !

    函數返回值在調用該函數後通過 $? 來獲得

    注意:所有函數在使用前必須定義。這意味着必須將函數放在腳本開始部分,直至shell解釋器首次發現它時,纔可以使用。調用函數僅使用其函數名即可。

    函數參數

    在Shell中,調用函數時可以向其傳遞參數。在函數體內部,通過 n 1表示第一個參數,$2表示第二個參數…
    帶參數的函數示例:

    #!/bin/bash
    
    funWithParam(){
        echo "第一個參數爲 $1 !"
        echo "第二個參數爲 $2 !"
        echo "第十個參數爲 $10 !"
        echo "第十個參數爲 ${10} !"
        echo "第十一個參數爲 ${11} !"
        echo "參數總數有 $# 個!"
        echo "作爲一個字符串輸出所有參數 $* !"
    }
    funWithParam 1 2 3 4 5 6 7 8 9 34 73

    輸出結果:

    第一個參數爲 1 !
    第二個參數爲 2 !
    第十個參數爲 10 !
    第十個參數爲 34 !
    第十一個參數爲 73 !
    參數總數有 11!
    作爲一個字符串輸出所有參數 1 2 3 4 5 6 7 8 9 34 73 !

    注意,10 {10}。當n>=10時,需要使用${n}來獲取參數。

    另外,還有幾個特殊字符用來處理參數:
    這裏寫圖片描述

    Shell 輸入/輸出重定向

    大多數 UNIX 系統命令從你的終端接受輸入並將所產生的輸出發送回​​到您的終端。一個命令通常從一個叫標準輸入的地方讀取輸入,默認情況下,這恰好是你的終端。同樣,一個命令通常將其輸出寫入到標準輸出,默認情況下,這也是你的終端。

    重定向命令列表如下:
    這裏寫圖片描述

    需要注意的是文件描述符 0 通常是標準輸入(STDIN),1 是標準輸出(STDOUT),2 是標準錯誤輸出(STDERR)。

    輸出重定向

    重定向一般通過在命令間插入特定的符號來實現。特別的,這些符號的語法如下所示:

    command1 > file1

    上面這個命令執行command1然後將輸出的內容存入file1。

    注意任何file1內的已經存在的內容將被新內容替代。如果要將新內容添加在文件末尾,請使用>>操作符。

    實例

    執行下面的 who 命令,它將命令的完整的輸出重定向在用戶文件中(users):

    $ who > users

    執行後,並沒有在終端輸出信息,這是因爲輸出已被從默認的標準輸出設備(終端)重定向到指定的文件。

    你可以使用 cat 命令查看文件內容:

    $ cat users
    _mbsetupuser console  Oct 31 17:35 
    tianqixin    console  Oct 31 17:35 
    tianqixin    ttys000  Dec  1 11:33 

    輸出重定向會覆蓋文件內容,請看下面的例子:

    $ echo "菜鳥教程:www.runoob.com" > users
    $ cat users
    菜鳥教程:www.runoob.com
    $

    如果不希望文件內容被覆蓋,可以使用 >> 追加到文件末尾,例如:

    $ echo "菜鳥教程:www.runoob.com" >> users
    $ cat users
    菜鳥教程:www.runoob.com
    菜鳥教程:www.runoob.com
    $

    輸入重定向

    和輸出重定向一樣,Unix 命令也可以從文件獲取輸入,語法爲:

    command1 < file1

    這樣,本來需要從鍵盤獲取輸入的命令會轉移到文件讀取內容。
    注意:輸出重定向是大於號(>),輸入重定向是小於號(<)。

    實例

    接着以上實例,我們需要統計 users 文件的行數,執行以下命令:

    $ wc -l users
           2 users

    也可以將輸入重定向到 users 文件:

    $  wc -l < users
           2 

    注意:上面兩個例子的結果不同:第一個例子,會輸出文件名;第二個不會,因爲它僅僅知道從標準輸入讀取內容。

    command1 < infile > outfile

    同時替換輸入和輸出,執行command1,從文件infile讀取內容,然後將輸出寫入到outfile中。

    重定向深入講解

    一般情況下,每個 Unix/Linux 命令運行時都會打開三個文件:

  • 標準輸入文件(stdin):stdin的文件描述符爲0,Unix程序默認從stdin讀取數據。
  • 標準輸出文件(stdout):stdout 的文件描述符爲1,Unix程序默認向stdout輸出數據。
  • 標準錯誤文件(stderr):stderr的文件描述符爲2,Unix程序會向stderr流中寫入錯誤信息。
  • 默認情況下,command > file 將 stdout 重定向到 file,command < file 將stdin 重定向到 file。

    如果希望 stderr 重定向到 file,可以這樣寫:

    $ command 2 > file

    如果希望 stderr 追加到 file 文件末尾,可以這樣寫:

    $ command 2 >> file

    2 表示標準錯誤文件(stderr)。

    如果希望將 stdout 和 stderr 合併後重定向到 file,可以這樣寫:

    $ command > file 2>&1
    或者
    $ command >> file 2>&1

    如果希望對 stdin 和 stdout 都重定向,可以這樣寫:

    $ command < file1 >file2

    command 命令將 stdin 重定向到 file1,將 stdout 重定向到 file2。

    Here Document

    Here Document 是 Shell 中的一種特殊的重定向方式,用來將輸入重定向到一個交互式 Shell 腳本或程序。

    它的基本的形式如下:

    command << delimiter
        document
    delimiter

    它的作用是將兩個 delimiter 之間的內容(document) 作爲輸入傳遞給 command。

    注意:

  • 結尾的delimiter 一定要頂格寫,前面不能有任何字符,後面也不能有任何字符,包括空格和 tab 縮進。
  • 開始的delimiter前後的空格會被忽略掉。
  • 實例

    在命令行中通過 wc -l 命令計算 Here Document 的行數:

    $ wc -l << EOF
        歡迎來到
        菜鳥教程
        www.runoob.com
    EOF
    3          # 輸出結果爲 3 行
    $

    我們也可以將 Here Document 用在腳本中,例如:

    #!/bin/bash
    
    cat << EOF
    歡迎來到
    菜鳥教程
    www.runoob.com
    EOF

    執行以上腳本,輸出結果:

    歡迎來到
    菜鳥教程
    www.runoob.com

    /dev/null 文件

    如果希望執行某個命令,但又不希望在屏幕上顯示輸出結果,那麼可以將輸出重定向到 /dev/null:

    $ command > /dev/null

    /dev/null 是一個特殊的文件,寫入到它的內容都會被丟棄;如果嘗試從該文件讀取內容,那麼什麼也讀不到。但是 /dev/null 文件非常有用,將命令的輸出重定向到它,會起到”禁止輸出”的效果。

    如果希望屏蔽 stdout 和 stderr,可以這樣寫:

    $ command > /dev/null 2>&1

    注意:0 是標準輸入(STDIN),1 是標準輸出(STDOUT),2 是標準錯誤輸出(STDERR)。

    Shell 文件包含

    和其他語言一樣,Shell 也可以包含外部腳本。這樣可以很方便的封裝一些公用的代碼作爲一個獨立的文件。

    Shell 文件包含的語法格式如下:

    . filename   # 注意點號(.)和文件名中間有一空格source filename

    實例

    創建兩個 shell 腳本文件。

    test1.sh 代碼如下:

    #!/bin/bash
    
    url="http://www.runoob.com"

    test2.sh 代碼如下:

    #!/bin/bash
    
    #使用 . 號來引用test1.sh 文件
    . ./test1.sh
    
    # 或者使用以下包含文件代碼
    # source ./test1.sh
    
    echo "菜鳥教程官網地址:$url"

    接下來,我們爲 test2.sh 添加可執行權限並執行:

    $ chmod +x test2.sh 
    $ ./test2.sh 
    菜鳥教程官網地址:http://www.runoob.com

    注:被包含的文件 test1.sh 不需要可執行權限。

    發表評論
    所有評論
    還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
    相關文章