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;]
}
說明:
下面的例子定義了一個函數並進行調用:
#!/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
兩個數字分別爲 1 和 2 !
輸入的兩個數字之和爲 3 !
函數返回值在調用該函數後通過 $? 來獲得。
注意:所有函數在使用前必須定義。這意味着必須將函數放在腳本開始部分,直至shell解釋器首次發現它時,纔可以使用。調用函數僅使用其函數名即可。
函數參數
在Shell中,調用函數時可以向其傳遞參數。在函數體內部,通過
帶參數的函數示例:
#!/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 !
注意,
另外,還有幾個特殊字符用來處理參數:
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 命令運行時都會打開三個文件:
默認情況下,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。
注意:
實例
在命令行中通過 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 不需要可執行權限。