一、小括號,圓括號()
<a name="t1" style="box-sizing: border-box; background: transparent; color: rgb(79, 161, 219); text-decoration: none; margin: 0px; padding: 0px; font-weight: 400; outline: 0px;"></a>
1、單小括號 ()
①命令組。括號中的命令將會新開一個子shell順序執(zhí)行,所以括號中的變量不能夠被腳本余下的部分使用。括號中多個命令之間用分號隔開,最后一個命令可以沒有分號,各命令和括號之間不必有空格。
②命令替換。等同于cmd
,shell掃描一遍命令行,發(fā)現(xiàn)了$(cmd)結(jié)構(gòu),便將$(cmd)中的cmd執(zhí)行一次,得到其標(biāo)準(zhǔn)輸出,再將此輸出放到原來命令。有些shell不支持,如tcsh。
③用于初始化數(shù)組。如:array=(a b c d)
<a name="t2" style="box-sizing: border-box; background: transparent; color: rgb(79, 161, 219); text-decoration: none; margin: 0px; padding: 0px; font-weight: 400; outline: 0px;"></a>
2、雙小括號 (( ))
①整數(shù)擴(kuò)展。這種擴(kuò)展計(jì)算是整數(shù)型的計(jì)算,不支持浮點(diǎn)型。((exp))結(jié)構(gòu)擴(kuò)展并計(jì)算一個算術(shù)表達(dá)式的值,如果表達(dá)式的結(jié)果為0,那么返回的退出狀態(tài)碼為1,或者 是"假",而一個非零值的表達(dá)式所返回的退出狀態(tài)碼將為0,或者是"true"。若是邏輯判斷,表達(dá)式exp為真則為1,假則為0。
②只要括號中的運(yùn)算符、表達(dá)式符合C語言運(yùn)算規(guī)則,都可用在$((exp))中,甚至是三目運(yùn)算符。作不同進(jìn)位(如二進(jìn)制、八進(jìn)制、十六進(jìn)制)運(yùn)算時,輸出結(jié)果全都自動轉(zhuǎn)化成了十進(jìn)制。如:echo $((16#5f)) 結(jié)果為95 (16進(jìn)位轉(zhuǎn)十進(jìn)制)
③單純用 (( )) 也可重定義變量值,比如 a=5; ((a++)) 可將 $a 重定義為6
④常用于算術(shù)運(yùn)算比較,雙括號中的變量可以不使用$符號前綴。括號內(nèi)支持多個表達(dá)式用逗號分開。
只要括號中的表達(dá)式符合C語言運(yùn)算規(guī)則,
比如可以直接使用for((i=0;i<5;i++)), 如果不使用雙括號, 則為for i in seq 0 4
或者for i in {0..4}。
再如可以直接使用if (($i<5)), 如果不使用雙括號, 則為if [ $i -lt 5 ]。
<a name="t3" style="box-sizing: border-box; background: transparent; color: rgb(79, 161, 219); text-decoration: none; margin: 0px; padding: 0px; font-weight: 400; outline: 0px;"></a>
二、中括號,方括號[]
<a name="t4" style="box-sizing: border-box; background: transparent; color: rgb(79, 161, 219); text-decoration: none; margin: 0px; padding: 0px; font-weight: 400; outline: 0px;"></a>
1、單中括號 []
①bash 的內(nèi)部命令,[和test是等同的。如果我們不用絕對路徑指明,通常我們用的都是bash自帶的命令。if/test結(jié)構(gòu)中的左中括號是調(diào)用test的命令標(biāo)識,右中括號是關(guān)閉條件判斷的。這個命令把它的參數(shù)作為比較表達(dá)式或者作為文件測試,并且根據(jù)比較的結(jié)果來返回一個退出狀態(tài)碼。if/test結(jié)構(gòu)中并不是必須右中括號,但是新版的Bash中要求必須這樣。
②Test和[]中可用的比較運(yùn)算符只有==和!=,兩者都是用于字符串比較的,不可用于整數(shù)比較,整數(shù)比較只能使用-eq,-gt這種形式。無論是字符串比較還是整數(shù)比較都不支持大于號小于號。如果實(shí)在想用,對于字符串比較可以使用轉(zhuǎn)義形式,如果比較"ab"和"bc":[ ab \< bc ],結(jié)果為真,也就是返回狀態(tài)為0。[ ]中的邏輯與和邏輯或使用-a 和-o 表示。
③字符范圍。用作正則表達(dá)式的一部分,描述一個匹配的字符范圍。作為test用途的中括號內(nèi)不能使用正則。
④在一個array 結(jié)構(gòu)的上下文中,中括號用來引用數(shù)組中每個元素的編號。
<a name="t5" style="box-sizing: border-box; background: transparent; color: rgb(79, 161, 219); text-decoration: none; margin: 0px; padding: 0px; font-weight: 400; outline: 0px;"></a>
2、雙中括號[[ ]]
①[[是 bash 程序語言的關(guān)鍵字。并不是一個命令,[[ ]] 結(jié)構(gòu)比[ ]結(jié)構(gòu)更加通用。在[[和]]之間所有的字符都不會發(fā)生文件名擴(kuò)展或者單詞分割,但是會發(fā)生參數(shù)擴(kuò)展和命令替換。
②支持字符串的模式匹配,使用=~操作符時甚至支持shell的正則表達(dá)式。字符串比較時可以把右邊的作為一個模式,而不僅僅是一個字符串,比如[[ hello == hell? ]],結(jié)果為真。[[ ]] 中匹配字符串或通配符,不需要引號。
③使用[[ ... ]]條件判斷結(jié)構(gòu),而不是[ ... ],能夠防止腳本中的許多邏輯錯誤。比如,&&、||、<和> 操作符能夠正常存在于[[ ]]條件判斷結(jié)構(gòu)中,但是如果出現(xiàn)在[ ]結(jié)構(gòu)中的話,會報(bào)錯。
比如可以直接使用if [[ $a != 1 && $a != 2 ]], 如果不適用雙括號, 則為if [ $a -ne 1] && [ $a != 2 ]或者if [ $a -ne 1 -a $a != 2 ]。
④bash把雙中括號中的表達(dá)式看作一個單獨(dú)的元素,并返回一個退出狀態(tài)碼。
例子:
[cpp]
<embed id="ZeroClipboardMovie_1" src="http://static.blog.csdn.net/scripts/ZeroClipboard/ZeroClipboard.swf" loop="false" menu="false" quality="best" bgcolor="#ffffff" width="16" height="16" name="ZeroClipboardMovie_1" align="middle" allowscriptaccess="always" allowfullscreen="false" type="application/x-shockwave-flash" pluginspage="http://www.macromedia.com/go/getflashplayer" flashvars="id=1&width=16&height=16" wmode="transparent" style="box-sizing: border-box;">
-
if
($i<5)
-
if
[ $i -lt 5 ]
-
if
[ $a -ne 1 -a $a != 2 ]
-
if
[ $a -ne 1] && [ $a != 2 ]
-
if
[[ $a != 1 && $a != 2 ]]
-
for
i in $(seq 0 4);
do
echo $i;done
-
for
i in
seq 0 4
;do
echo $i;done
-
for
((i=0;i<5;i++));
do
echo $i;done
-
for
i in {0..4};
do
echo $i;done
<a name="t6" style="box-sizing: border-box; background: transparent; color: rgb(79, 161, 219); text-decoration: none; margin: 0px; padding: 0px; font-weight: 400; outline: 0px;"></a>
三、大括號、花括號 {}
<a name="t7" style="box-sizing: border-box; background: transparent; color: rgb(79, 161, 219); text-decoration: none; margin: 0px; padding: 0px; font-weight: 400; outline: 0px;"></a>
1、常規(guī)用法
①大括號拓展。(通配(globbing))將對大括號中的文件名做擴(kuò)展。在大括號中,不允許有空白,除非這個空白被引用或轉(zhuǎn)義。第一種:對大括號中的以逗號分割的文件列表進(jìn)行拓展。如 touch {a,b}.txt 結(jié)果為a.txt b.txt。第二種:對大括號中以點(diǎn)點(diǎn)(..)分割的順序文件列表起拓展作用,如:touch {a..d}.txt 結(jié)果為a.txt b.txt c.txt d.txt
[cpp]
<embed id="ZeroClipboardMovie_2" src="http://static.blog.csdn.net/scripts/ZeroClipboard/ZeroClipboard.swf" loop="false" menu="false" quality="best" bgcolor="#ffffff" width="16" height="16" name="ZeroClipboardMovie_2" align="middle" allowscriptaccess="always" allowfullscreen="false" type="application/x-shockwave-flash" pluginspage="http://www.macromedia.com/go/getflashplayer" flashvars="id=2&width=16&height=16" wmode="transparent" style="box-sizing: border-box;">
-
ls {ex1,ex2}.sh
ex1.sh ex2.sh
-
ls {ex{1..3},ex4}.sh
ex1.sh ex2.sh ex3.sh ex4.sh
-
ls {ex[1-3],ex4}.sh
ex1.sh ex2.sh ex3.sh ex4.sh
②代碼塊,又被稱為內(nèi)部組,這個結(jié)構(gòu)事實(shí)上創(chuàng)建了一個匿名函數(shù) 。與小括號中的命令不同,大括號內(nèi)的命令不會新開一個子shell運(yùn)行,即腳本余下部分仍可使用括號內(nèi)變量。括號內(nèi)的命令間用分號隔開,最后一個也必須有分號。{}的第一個命令和左括號之間必須要有一個空格。
<a name="t8" style="box-sizing: border-box; background: transparent; color: rgb(79, 161, 219); text-decoration: none; margin: 0px; padding: 0px; font-weight: 400; outline: 0px;"></a>
2、幾種特殊的替換結(jié)構(gòu)
<a name="t9" style="box-sizing: border-box; background: transparent; color: rgb(79, 161, 219); text-decoration: none; margin: 0px; padding: 0px; font-weight: 400; outline: 0px;"></a>
${var:-string},${var:+string},${var:=string},${var:?string}
①${var:-string}和${var:=string}:若變量var為空,則用在命令行中用string來替換${var:-string},否則變量var不為空時,則用變量var的值來替換${var:-string};對于${var:=string}的替換規(guī)則和${var:-string}是一樣的,所不同之處是${var:=string}若var為空時,用string替換${var:=string}的同時,把string賦給變量var: ${var:=string}很常用的一種用法是,判斷某個變量是否賦值,沒有的話則給它賦上一個默認(rèn)值。
② ${var:+string}的替換規(guī)則和上面的相反,即只有當(dāng)var不是空的時候才替換成string,若var為空時則不替換或者說是替換成變量 var的值,即空值。(因?yàn)樽兞縱ar此時為空,所以這兩種說法是等價(jià)的)
③${var:?string}替換規(guī)則為:若變量var不為空,則用變量var的值來替換${var:?string};若變量var為空,則把string輸出到標(biāo)準(zhǔn)錯誤中,并從腳本中退出。我們可利用此特性來檢查是否設(shè)置了變量的值。
補(bǔ)充擴(kuò)展:在上面這五種替換結(jié)構(gòu)中string不一定是常值的,可用另外一個變量的值或是一種命令的輸出。
<a name="t10" style="box-sizing: border-box; background: transparent; color: rgb(79, 161, 219); text-decoration: none; margin: 0px; padding: 0px; font-weight: 400; outline: 0px;"></a>
3、四種模式匹配替換結(jié)構(gòu)
模式匹配記憶方法:
是去掉左邊(在鍵盤上#在$之左邊)
% 是去掉右邊(在鍵盤上%在$之右邊)
和%中的單一符號是最小匹配,兩個相同符號是最大匹配。
<a name="t11" style="box-sizing: border-box; background: transparent; color: rgb(79, 161, 219); text-decoration: none; margin: 0px; padding: 0px; font-weight: 400; outline: 0px;"></a>
${var%pattern},${var%%pattern},${var#pattern},${var##pattern}
第一種模式:${variable%pattern},這種模式時,shell在variable中查找,看它是否一給的模式pattern結(jié)尾,如果是,就從命令行把variable中的內(nèi)容去掉右邊最短的匹配模式
第二種模式: ${variable%%pattern},這種模式時,shell在variable中查找,看它是否一給的模式pattern結(jié)尾,如果是,就從命令行把variable中的內(nèi)容去掉右邊最長的匹配模式
第三種模式:${variable#pattern} 這種模式時,shell在variable中查找,看它是否一給的模式pattern開始,如果是,就從命令行把variable中的內(nèi)容去掉左邊最短的匹配模式
第四種模式: ${variable##pattern} 這種模式時,shell在variable中查找,看它是否一給的模式pattern結(jié)尾,如果是,就從命令行把variable中的內(nèi)容去掉右邊最長的匹配模式
這四種模式中都不會改變variable的值,其中,只有在pattern中使用了*匹配符號時,%和%%,#和##才有區(qū)別。結(jié)構(gòu)中的pattern支持通配符,*表示零個或多個任意字符,?表示僅與一個任意字符匹配,[...]表示匹配中括號里面的字符,[!...]表示不匹配中括號里面的字符
。
[cpp]
<embed id="ZeroClipboardMovie_3" src="http://static.blog.csdn.net/scripts/ZeroClipboard/ZeroClipboard.swf" loop="false" menu="false" quality="best" bgcolor="#ffffff" width="16" height="16" name="ZeroClipboardMovie_3" align="middle" allowscriptaccess="always" allowfullscreen="false" type="application/x-shockwave-flash" pluginspage="http://www.macromedia.com/go/getflashplayer" flashvars="id=3&width=16&height=16" wmode="transparent" style="box-sizing: border-box;">
-
var=testcase
-
echo $var
testcase
-
echo ${var%s*e}
testca
-
echo $var
testcase
-
echo ${var%%s*e}
te
-
echo ${var#?e}
stcase
-
echo ${var##?e}
stcase
-
echo ${var##*e}
-
echo ${var##*s}
e
-
echo ${var##test}
case
<a name="t12" style="box-sizing: border-box; background: transparent; color: rgb(79, 161, 219); text-decoration: none; margin: 0px; padding: 0px; font-weight: 400; outline: 0px;"></a>
4、字符串提取和替換
<a name="t13" style="box-sizing: border-box; background: transparent; color: rgb(79, 161, 219); text-decoration: none; margin: 0px; padding: 0px; font-weight: 400; outline: 0px;"></a>
${var:num},${var
:num
1:num2},${var/
pattern
/pattern},${var//pattern/pattern}
第一種模式:${var:num},這種模式時,shell在
var
中提取第num個字符到末尾的所有字符。若num為正數(shù),從左邊0處開始;若num為負(fù)數(shù),從右邊開始提取字串,但必須使用在冒號后面加空格或一個數(shù)字或整個num加上括號,如${var: -2}、${var:1-3}或${var:(-2)}。
<font size="3" style="box-sizing: border-box;"><font size="3" style="box-sizing: border-box;">第二種模式:${var:num1:num2},num1是位置,num2是長度。表示從$var字符串的第$num1個位置開始提取長度為$num2的子串。不能為負(fù)數(shù)。
第
三種模式:${var/pattern/pattern}表示將var字符串的第一個匹配的pattern替換為另一個pattern。</font> <font size="3" style="box-sizing: border-box;"><font size="3" style="box-sizing: border-box;"><font size="3" style="box-sizing: border-box;"><font size="3" style="box-sizing: border-box;"><font size="3" style="box-sizing: border-box;"><font size="3" style="box-sizing: border-box;">。<font size="3" style="box-sizing: border-box;">第
四種模式:${var//pattern/pattern}表示將var字符串中的所有能匹配的pattern替換為另一個pattern。</font></font></font></font></font></font></font></font>
[html]
<embed id="ZeroClipboardMovie_4" src="http://static.blog.csdn.net/scripts/ZeroClipboard/ZeroClipboard.swf" loop="false" menu="false" quality="best" bgcolor="#ffffff" width="16" height="16" name="ZeroClipboardMovie_4" align="middle" allowscriptaccess="always" allowfullscreen="false" type="application/x-shockwave-flash" pluginspage="http://www.macromedia.com/go/getflashplayer" flashvars="id=4&width=16&height=16" wmode="transparent" style="box-sizing: border-box;">
-
[root@centos ~]#
var
=/home/centos
[root@centos ~]# echo $var
/home/centos
[root@centos ~]# echo ${var:5}
/centos
[root@centos ~]# echo ${var: -6}
centos
[root@centos ~]# echo ${var:(-6)}
centos
[root@centos ~]# echo ${var:1:4}
home
[root@centos ~]# echo ${var/o/h}
/hhme/centos
[root@centos ~]# echo ${var//o/h}
/hhme/cenths
<a name="t14" style="box-sizing: border-box; background: transparent; color: rgb(79, 161, 219); text-decoration: none; margin: 0px; padding: 0px; font-weight: 400; outline: 0px;"></a>
四、符號$后的括號
(1)${a} 變量a的值, 在不引起歧義的情況下可以省略大括號。
(2)$(cmd) 命令替換,和cmd
效果相同,結(jié)果為shell命令cmd的輸,過某些Shell版本不支持$()形式的命令替換, 如tcsh。
(3)$((expression)) 和exprexpression
效果相同, 計(jì)算數(shù)學(xué)表達(dá)式exp的數(shù)值, 其中exp只要符合C語言的運(yùn)算規(guī)則即可, 甚至三目運(yùn)算符和邏輯表達(dá)式都可以計(jì)算。
<a name="t15" style="box-sizing: border-box; background: transparent; color: rgb(79, 161, 219); text-decoration: none; margin: 0px; padding: 0px; font-weight: 400; outline: 0px;"></a>
五、使用
1、
多條命令執(zhí)行
(1)單小括號,
(cmd1;cmd2;cmd3) 新開一個子shell順序執(zhí)行命令cmd1,cmd2,cmd3, 各命令之間用分號隔開, 最后一個命令后可以沒有分號
。
(2)單大括號,
{ cmd1;cmd2;cmd3;} 在當(dāng)前shell順序執(zhí)行命令cmd1,cmd2,cmd3, 各命令之間用分號隔開, 最后一個命令后必須有分號, 第一條命令和左括號之間必須用空格隔開。
對{}和()而言, 括號中的重定向符只影響該條命令, 而括號外的重定向符影響到括號中的所有命令。