Shell程序設(shè)計(jì)的流程控制
和其他高級(jí)程序設(shè)計(jì)語(yǔ)言一樣,Shell提供了用來(lái)控制程序執(zhí)行流程的命令,包括條件分支和循環(huán)結(jié)構(gòu),用戶可以用這些命令創(chuàng)建非常復(fù)雜的程序。
與傳統(tǒng)語(yǔ)言不同的是,Shell用于指定條件值的不是布爾運(yùn)算式,而是命令和字串。
1.測(cè)試命令
test命令用于檢查某個(gè)條件是否成立,它可以進(jìn)行數(shù)值、字符和文件3個(gè)方面的測(cè)試,其測(cè)試符和相應(yīng)的功能分別如下。
(1)數(shù)值測(cè)試:
-eq 等于則為真。
-ne 不等于則為真。
-gt 大于則為真。
-ge 大于等于則為真。
-lt 小于則為真。
-le 小于等于則為真。
(2)字串測(cè)試:
= 等于則為真。
!= 不相等則為真。
-z字串 字串長(zhǎng)度偽則為真。
-n字串 字串長(zhǎng)度不偽則為真。
(3)文件測(cè)試:
-e文件名 如果文件存在則為真。
-r文件名 如果文件存在且可讀則為真。
-w文件名 如果文件存在且可寫則為真。
-x文件名 如果文件存在且可執(zhí)行則為真。
-s文件名 如果文件存在且至少有一個(gè)字符則為真。
-d文件名 如果文件存在且為目錄則為真。
-f文件名 如果文件存在且為普通文件則為真。
-c文件名 如果文件存在且為字符型特殊文件則為真。
-b文件名 如果文件存在且為塊特殊文件則為真。
另外,Linux還提供了與(?。⒒颍?o)、非(-a)三個(gè)邏輯操作符,用于將測(cè)試條件連接起來(lái),其優(yōu)先順序?yàn)椋?!最高?a次之,-o最低。
同時(shí),bash也能完成簡(jiǎn)單的算術(shù)運(yùn)算,格式如下:
$[expression]
例如:
var1=2
var2=$[var1*10+1]
則var2的值為21。
2.if條件語(yǔ)句
Shell程序中的條件分支是通過(guò)if條件語(yǔ)句來(lái)實(shí)現(xiàn)的,其一般格式為:
if 條件命令串
then
條件為真時(shí)的命令串
else
條件為假時(shí)的命令串
fi
3.for循環(huán)
for循環(huán)對(duì)一個(gè)變量的可能的值都執(zhí)行一個(gè)命令序列。賦給變量的幾個(gè)數(shù)值既可以在程序內(nèi)以數(shù)值列表的形式提供,也可以在程序以外以位置參數(shù)的形式提供。for循環(huán)的一般格式為:
for變量名[in數(shù)值列表]
do
若干個(gè)命令行
done
變量名可以是用戶選擇的任何字串,如果變量名是var,則在in之后給出的數(shù)值將順序替換循環(huán)命令列表中的$var。如果省略了in,則變量var的取值將是位置參數(shù)。對(duì)變量的每一個(gè)可能的賦值都將執(zhí)行do和done之間的命令列表。
4.while和until循環(huán)
while和until命令都是用命令的返回狀態(tài)值來(lái)控制循環(huán)的。While循環(huán)的一般格式為:
while
若干個(gè)命令行1
do
若干個(gè)命令行2
done
只要while的“若干個(gè)命令行1”中最后一個(gè)命令的返回狀態(tài)為真,while循環(huán)就繼續(xù)執(zhí)行do...done之間的“若干個(gè)命令行2”。
until命令是另一種循環(huán)結(jié)構(gòu),它和while命令相似,其格式如下:
until
若干個(gè)命令行1
do
若干個(gè)命令行2
done
until循環(huán)和while循環(huán)的區(qū)別在于:while循環(huán)在條件為真時(shí)繼續(xù)執(zhí)行循環(huán),而until則是在條件為假時(shí)繼續(xù)執(zhí)行循環(huán)。
Shell還提供了true和false兩條命令用于創(chuàng)建無(wú)限循環(huán)結(jié)構(gòu),它們的返回狀態(tài)分別是總為0或總為非0。
5.case條件選擇
if條件語(yǔ)句用于在兩個(gè)選項(xiàng)中選定一項(xiàng),而case條件選擇為用戶提供了根據(jù)字串或變量的值從多個(gè)選項(xiàng)中選擇一項(xiàng)的方法,其格式如下:
case string in
exp-1)
若干個(gè)命令行1
;;
exp-2)
若干個(gè)命令行2
;;
……
*)
其他命令行
esac
Shell通過(guò)計(jì)算字串string的值,將其結(jié)果依次和運(yùn)算式exp-1, exp-2等進(jìn)行比較,直到找到一個(gè)匹配的運(yùn)算式為止。如果找到了匹配項(xiàng),則執(zhí)行它下面的命令直到遇到一對(duì)分號(hào)(;;)為止。
在case運(yùn)算式中也可以使用Shell的通配符(“*”、“?”、“[ ]”)。通常用 * 作為case命令的最后運(yùn)算式以便在前面找不到任何相應(yīng)的匹配項(xiàng)時(shí)執(zhí)行“其他命令行”的命令。
6.無(wú)條件控制語(yǔ)句break和continue
break用于立即終止當(dāng)前循環(huán)的執(zhí)行,而contiune用于不執(zhí)行循環(huán)中后面的語(yǔ)句而立即開(kāi)始下一個(gè)循環(huán)的執(zhí)行。這兩個(gè)語(yǔ)句只有放在do和done之間才有效。
7.函數(shù)定義
在Shell中還可以定義函數(shù)。函數(shù)實(shí)際上也是由若干條Shell命令組成的,因此它與Shell程序形式上是相似的,不同的是它不是一個(gè)單獨(dú)的進(jìn)程,而是Shell程序的一部分。函數(shù)定義的基本格式為:
functionname
{
若干命令行
}
調(diào)用函數(shù)的格式為:
functionname param1 param2…
Shell函數(shù)可以完成某些例行的工作,而且還可以有自己的退出狀態(tài),因此函數(shù)也可以作為if, while等控制結(jié)構(gòu)的條件。
在函數(shù)定義時(shí)不用帶參數(shù)說(shuō)明,但在調(diào)用函數(shù)時(shí)可以帶有參數(shù),此時(shí)Shell將把這些參數(shù)分別賦予相應(yīng)的位置參數(shù)$1, $2, ...及$*。
8.命令分組
在Shell中有兩種命令分組的方法:()和{}。前者當(dāng)Shell執(zhí)行()中的命令時(shí)將再創(chuàng)建一個(gè)新的子進(jìn)程,然后這個(gè)子進(jìn)程去執(zhí)行圓括弧中的命令。當(dāng)用戶在執(zhí)行某個(gè)命令時(shí)不想讓命令運(yùn)行時(shí)對(duì)狀態(tài)集合(如位置參數(shù)、環(huán)境變量、當(dāng)前工作目錄等)的改變影響到下面語(yǔ)句的執(zhí)行時(shí),就應(yīng)該把這些命令放在圓括弧中,這樣就能保證所有的改變只對(duì)子進(jìn)程產(chǎn)生影響,而父進(jìn)程不受任何干擾。{}用于將順序執(zhí)行的命令的輸出結(jié)果用于另一個(gè)命令的輸入(管道方式)。當(dāng)我們要真正使用圓括弧和花括弧時(shí)(如計(jì)算運(yùn)算式的優(yōu)先順序),則需要在其前面加上轉(zhuǎn)義符()以便讓Shell知道它們不是用于命令執(zhí)行的控制所用。
9.信號(hào)
trap命令用于在Shell程序中捕捉信號(hào),之后可以有3種反應(yīng)方式:
(1)執(zhí)行一段程序來(lái)處理這一信號(hào)。
(2)接受信號(hào)的默認(rèn)操作。
(3)忽視這一信號(hào)。
trap對(duì)上面3種方式提供了3種基本形式:
第一種形式的trap命令在Shell接收到與signal list清單中數(shù)值相同的信號(hào)時(shí),將執(zhí)行雙引號(hào)中的命令串。
trap 'commands' signal-list
trap "commands" signal-list
為了恢復(fù)信號(hào)的默認(rèn)操作,使用第二種形式的trap命令:
trap signal-list
第三種形式的trap命令允許忽略信號(hào):
trap " " signal-list
注意:
(1)對(duì)信號(hào)11(段違例)不能捕捉,因?yàn)镾hell本身需要捕捉該信號(hào)去進(jìn)行內(nèi)存的轉(zhuǎn)儲(chǔ)。
(2)在trap中可以定義對(duì)信號(hào)0的處理(實(shí)際上沒(méi)有這個(gè)信號(hào)),Shell程序在其終止(如執(zhí)行exit語(yǔ)句)時(shí)發(fā)出該信號(hào)。
(3)在捕捉到signal-list中指定的信號(hào)并執(zhí)行完相應(yīng)的命令之后,如果這些命令沒(méi)有將Shell程序終止的話,Shell程序?qū)⒗^續(xù)執(zhí)行收到信號(hào)時(shí)所執(zhí)行的命令后面的命令,這樣將很容易導(dǎo)致Shell程序無(wú)法終止。
另外,在trap語(yǔ)句中,單引號(hào)和雙引號(hào)是不同的。當(dāng)Shell程序第一次碰到trap語(yǔ)句時(shí),將把commands中的命令掃描一遍。此時(shí)若 commands是用單引號(hào)括起來(lái)的話,那么Shell不會(huì)對(duì)commands中的變量和命令進(jìn)行替換,否則commands中的變量和命令將用當(dāng)時(shí)具體的值來(lái)替換。