一、小括弧,園括弧()
1、單小括弧 () ①命令組。括弧中的命令將會新開一個子shell順序執行,所以括弧中的變數不能夠被指令碼餘下的部分使用。括弧中多個命令之間用分號隔開,最後一個命令可以沒有分號,各命令和括弧之間不必有空格。 ②命令替換。等同於`cmd`,shell掃描一遍命令列,發現了$(cmd)結構,便將$(cmd)中的cmd執行一次,得到其標準輸出,再將此輸出放到原來命令。有些shell不支援,如tcsh。 ③用於初始化數組。如:array=(a b c d)
2、雙小括弧 (( )) ①整數擴充。這種擴充計算是整數型的計算,不支援浮點型。((exp))結構擴充並計算一個算術運算式的值,如果運算式的結果為0,那麼返回的退出狀態代碼為1,或者 是"假",而一個非零值的運算式所返回的退出狀態代碼將為0,或者是"true"。若是邏輯判斷,運算式exp為真則為1,假則為0。 ②只要括弧中的運算子、運算式符合C語言運算規則,都可用在$((exp))中,甚至是三目運算子。作不同進位(如二進位、八進位、十六進位)運算時,輸出結果全都自動轉化成了十進位。如:echo $((16#5f)) 結果為95 (16進位轉十進位) ③單純用 (( )) 也可重定義變數值,比如 a=5; ((a++)) 可將 $a 重定義為6 ④雙括弧中的變數可以不使用$符號首碼。括弧內支援多個運算式用逗號分開。
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;donefor i in `seq 0 4`;do echo $i;donefor ((i=0;i<5;i++));do echo $i;donefor i in {0..4};do echo $i;done
二)中括弧,方括弧[]
1、單中括弧 [] ①bash 的內部命令,[和test是等同的。如果我們不用絕對路徑指明,通常我們用的都是bash內建的命令。if/test結構中的左中括弧是調用test的命令標識,右中括弧是關閉條件判斷的。這個命令把它的參數作為比較運算式或者作為檔案測試,並且根據比較的結果來返回一個退出狀態代碼。if/test結構中並不是必須右中括弧,但是新版的Bash中要求必須這樣。 ②Test和[]中可用的比較子只有==和!=,兩者都是用於字串比較的,不可用於整數比較,整數比較只能使用-eq,-gt這種形式。無論是字串比較還是整數比較都不支援大於符號小於符號。如果實在想用,對於字串比較可以使用轉義形式,如果比較"ab"和"bc":[ ab \< bc ],結果為真,也就是返回狀態為0。[ ]中的邏輯與和邏輯或使用-a 和-o 表示。 ③字元範圍。用作Regex的一部分,描述一個匹配的字元範圍。作為test用途的中括弧內不能使用正則。 ④在一個array 結構的上下文中,中括弧用來引用數組中每個元素的編號。
2、雙中括弧[[ ]] ①[[是 bash 程式語言的關鍵字。並不是一個命令,[[ ]] 結構比[ ]結構更加通用。在[[和]]之間所有的字元都不會發生檔案名稱擴充或者單詞分割,但是會發生參數擴充和命令替換。 ②支援字串的模式比對,使用=~操作符時甚至支援shell的Regex。字串比較時可以把右邊的作為一個模式,而不僅僅是一個字串,比如[[ hello == hell? ]],結果為真。[[ ]] 中匹配字串或萬用字元,不需要引號。 ③使用[[ ... ]]條件判斷結構,而不是[ ... ],能夠防止指令碼中的許多邏輯錯誤。比如,&&、||、<和> 操作符能夠正常存在於[[ ]]條件判斷結構中,但是如果出現在[ ]結構中的話,會報錯。 ④bash把雙中括弧中的運算式看作一個單獨的元素,並返回一個退出狀態代碼。
三)大括弧、花括弧 {}
1、常規用法。 ①大括弧拓展。(通配(globbing))將對大括弧中的檔案名稱做擴充。在大括弧中,不允許有空白,除非這個空白被引用或轉義。第一種:對大括弧中的以逗號分割的檔案清單進行拓展。如 touch {a,b}.txt 結果為a.txt b.txt。第二種:對大括弧中以點點(..)分割的循序檔列表起拓展作用,如:touch {a..d}.txt 結果為a.txt b.txt c.txt d.txt
bogon:/home/bash # ls {ex1,ex2}.shex1.sh ex2.shbogon:/home/bash # ls {ex{1..3},ex4}.shex1.sh ex2.sh ex3.sh ex4.shbogon:/home/bash # ls {ex[1-3],ex4}.shex1.sh ex2.sh ex3.sh ex4.sh
②代碼塊,又被稱為內部組,這個結構事實上建立了一個匿名函數 。與小括弧中的命令不同,大括弧內的命令不會新開一個子shell運行,即指令碼餘下部分仍可使用括弧內變數。括弧內的命令間用分號隔開,最後一個也必須有分號。{}的第一個命令和左括弧之間必須要有一個空格。
2)幾種特殊的替換結構:${var:-string},${var:+string},${var:=string},${var:?string} A,${var:-string}和${var:=string}:若變數var為空白,則用在命令列中用string來替換${var:-string},否則變數var不為空白時,則用變數var的值來替換${var:-string};對於${var:=string}的替換規則和${var:-string}是一樣的,所不同之處是${var:=string}若var為空白時,用string替換${var:=string}的同時,把string賦給變數var: ${var:=string}很常用的一種用法是,判斷某個變數是否賦值,沒有的話則給它賦上一個預設值。
B. ${var:+string}的替換規則和上面的相反,即只有當var不是空的時候才替換成string,若var為空白時則不替換或者說是替換成變數 var的值,即空值。(因為變數var此時為空白,所以這兩種說法是等價的)
C,${var:?string}替換規則為:若變數var不為空白,則用變數var的值來替換${var:?string};若變數var為空白,則把string輸出到標準錯誤中,並從指令碼中退出。我們可利用此特性來檢查是否設定了變數的值。
補充擴充:在上面這五種替換結構中string不一定是常值的,可用另外一個變數的值或是一種命令的輸出。
3)四種模式比對替換結構:${var%pattern},${var%%pattern},${var#pattern},${var##pattern} 第一種模式:${variable%pattern},這種模式時,shell在variable中尋找,看它是否一給的模式pattern結尾,如果是,就從命令列把variable中的內容去掉右邊最短的匹配模式
第二種模式: ${variable%%pattern},這種模式時,shell在variable中尋找,看它是否一給的模式pattern結尾,如果是,就從命令列把variable中的內容去掉右邊最長的匹配模式
第三種模式:${variable#pattern} 這種模式時,shell在variable中尋找,看它是否一給的模式pattern開始,如果是,就從命令列把variable中的內容去掉左邊最短的匹配模式
第四種模式: ${variable##pattern} 這種模式時,shell在variable中尋找,看它是否一給的模式pattern結尾,如果是,就從命令列把variable中的內容去掉右邊最長的匹配模式
這四種模式中都不會改變variable的值,其中,只有在pattern中使用了*匹配符號時,%和%%,#和##才有區別。結構中的pattern支援萬用字元,*表示零個或多個任一字元,?表示零個或一個任一字元,[...]表示匹配中括弧裡面的字元,[!...]表示不匹配中括弧裡面的字元
bogon:/home/bash # var=testcasebogon:/home/bash # echo $vartestcasebogon:/home/bash # echo ${var%s*e}testcabogon:/home/bash # echo $vartestcasebogon:/home/bash # echo ${var%%s*e}tebogon:/home/bash # echo ${var#?e}stcasebogon:/home/bash # echo ${var##?e}stcasebogon:/home/bash # echo ${var##*e}bogon:/home/bash # echo ${var##*s}ebogon:/home/bash # echo ${var##test}case