天天看點

shell-條件控制和函數

Shell 流程控制

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

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

在sh/bash裡可不能這麼寫,如果else分支沒有語句執行,就不要寫這個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 循環

一般格式:

for var in item1 item2 ... itemN
do
    command1
    command2
    ...
    commandN
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
           

或者

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
           

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腳本中可以随便調用。

格式:

[ function ] funname [()]
{
    action;
    [return int;]
}
           

說明: 可以帶function fun() 定義,也可以直接fun() 定義,不帶任何參數。 參數傳回,可以顯示加 return

傳回,若不加,将以最後一條指令運作結果,作為傳回值。return後跟數值n(0-255)

執行個體1(無return)

#!/bin/bash
myFun(){
    echo "This is my first shell function"
}
echo "-----function start-----"
myFun
echo "-----function start-----"
           

輸出

[email protected]-virtual-machine:~$ ./printf.sh 
-----func start-----
This is my first shell func
-----func start-----
           

執行個體2(有return)

#!/bin/bash

funWithReturn(){
    echo "Two nums add each other..."
    echo "Input the first num1:"
    read num1
    echo "Input the second num2:"
    read num2
    echo "Two nums are $num1 and $num2"
    return $(($num1+$num2))
}
funWithReturn
echo "num1 + num2 = $?"
           

輸出

[email protected]-virtual-machine:~$ ./printf.sh 
Two nums add each other...
Input the first num1:
1
Input the second num2:
5
Two nums are 1 and 5 !
num1 + num2 = 6
           

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

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

函數參數

在Shell中,調用函數時可以向其傳遞參數。在函數體内部,通過 $n 的形式來擷取參數的值,例如,$1表示第一個參數,$2表示第二個參數…

執行個體:

#!/bin/bash
funWithParam(){
    echo "The first param is $1"
    echo "The second param is $2"
    echo "The tenth param is $10"
    echo "The tenth param is ${10}"
    echo "The eleventh param is ${11}"
    echo "Total count is $# "
    echo "A string of all params is $*"
}
funWithParam 1 2 3 4 5 6 7 8 9 34 73
           

輸出

[email protected]-virtual-machine:~$ ./printf.sh 
first param is 1
second param is 2
tenth param is 10
tenth param is 34
eleventh param is 73
Total count is 11 
A string of all params is 1 2 3 4 5 6 7 8 9 34 73
           
注意:
$10 不能擷取第十個參數,擷取第十個參數需要${10}。當n>=10時,需要使用${n}來擷取參數。
           
幾個處理參數的特殊字元:
參數處理 說明
$# 傳遞到腳本的參數個數
$* 以一個單字元串顯示所有向腳本傳遞的參數
$$ 腳本運作的目前程序ID号
$! 背景運作的最後一個程序的ID号
[email protected] 與$*相同,但是使用時加引号,并在引号中傳回每個參數。
$- 顯示Shell使用的目前選項,與set指令功能相同。
$? 顯示最後指令的退出狀态。0表示沒有錯誤,其他任何值表明有錯誤。