轉至:https://www.jianshu.com/p/b88c7e07aaa9
linux中()、[]、{}、(())、[[]]等各種括號的使用
1、小括號、圓括號()
1.1 單小括號()
- 命令組。括號中的命令將會新開一個子shell順序執行,所以括號中的變量不能夠被腳本余下的部分使用。括號中多個命令之間用分號隔開,最后一個命令可以沒有分號,各命令和括 號之間不必有空格。
- 命令替換。等同於
cmd
,shell掃描一遍命令行,發現了$(cmd)結構,便將$(cmd)中的cmd執行一次,得到其標准輸出,再將此輸出放到原來命令。有些shell不支持,如tcsh。 - 初始化數組。如:array=(a b c d)。
1.2 雙小括號(())
- 整數擴展。這種擴展計算是整數型的計算,不支持浮點型。((exp))結構擴展並計算一個算術表達式的值,如果表達式的結果為0,那么返回的退出狀態碼為1,或者是"假",而一個非零值的表達式所返回的退出狀態碼將為0,或者是"true"。若是邏輯判斷,表達式exp為真則為1,假則為0。
- 只要括號中的運算符、表達式符合C語言運算規則,都可用在$((exp))中,甚至是三目運算符。作不同進位(如二進制、八進制、十六進制)運算時,輸出結果全都自動轉化成了十進制。如:echo $((16#5f)) 結果為95 (16進位轉十進制)
- 單純用 (( )) 也可重定義變量值,比如 a=5; ((a++)) 可將 $a 重定義為6。
- 常用於算術運算比較,雙括號中的變量可以不使用$符號前綴。括號內支持多個表達式用逗號分開。 只要括號中的表達式符合C語言運算規則,比如可以直接使用for((i=0;i<5;i++)), 如果不使用雙括號, 則為for i in
seq 0 4
或者for i in {0..4}。再如可以直接使用if (($i<5)), 如果不使用雙括號, 則為if [ $i -lt 5 ]。
1.3 簡單測試
[root@localhost ~]# for ((i=0;i<5;i++));do echo -n $i;done [root@localhost ~]# for i in $(seq 0 4);do echo -n $i;done [root@localhost ~]# for i in `seq 0 4`;do echo -n $i;done 01234
2、中括號、方括號[]
2.1 單中括號[]
- bash 的內部命令,[和test是等同的。如果我們不用絕對路徑指明,通常我們用的都是bash自帶的命令。if/test結構中的左中括號是調用test的命令標識,右中括號是關閉條件判斷的。這個命令把它的參數作為比較表達式或者作為文件測試,並且根據比較的結果來返回一個退出狀態碼。if/test結構中並不是必須右中括號,但是新版的Bash中要求必須這樣。
- test和[]中可用的比較運算符只有==和!=,兩者都是用於字符串比較的,不可用於整數比較,整數比較只能使用-eq,-gt這種形式。無論是字符串比較還是整數比較都不支持大於號小於號。如果實在想用,對於字符串比較可以使用轉義形式,如果比較"ab"和"bc":[ ab < bc ],結果為真,也就是返回狀態為0。[ ]中的邏輯與和邏輯或使用-a 和-o 表示。
- 字符范圍。用作正則表達式的一部分,描述一個匹配的字符范圍。作為test用途的中括號內不能使用正則。
- 在一個array 結構的上下文中,中括號用來引用數組中每個元素的編號。
2.2 雙中括號[[]]
- [[是 bash 程序語言的關鍵字。並不是一個命令,[[ ]] 結構比[ ]結構更加通用。在[[和]]之間所有的字符都不會發生文件名擴展或者單詞分割,但是會發生參數擴展和命令替換。
- 支持字符串的模式匹配,使用=~操作符時甚至支持shell的正則表達式。字符串比較時可以把右邊的作為一個模式,而不僅僅是一個字符串,比如[[ hello == hell? ]],結果為真。[[ ]] 中匹配字符串或通配符,不需要引號。
- 使用[[ ... ]]條件判斷結構,而不是[ ... ],能夠防止腳本中的許多邏輯錯誤。比如,&&、||、<和> 操作符能夠正常存在於[[ ]]條件判斷結構中,但是如果出現在[ ]結構中的話,會報錯。比如可以直接使用if [[ $a != 1 && $a != 2 ]], 如果不適用雙括號, 則為if [ $a -ne 1] && [ $a != 2 ]或者if [ $a -ne 1 -a $a != 2 ]。
- bash把雙中括號中的表達式看作一個單獨的元素,並返回一個退出狀態碼。
2.3 簡單測試
[root@localhost ~]# [[ 2\<3 ]] && echo true || false true [root@localhost ~]# [[ 2 -lt 3 ]] && echo true || false true [root@localhost ~]# [[ 2 \< 3 ]] && echo true || false -bash: 期待二元條件運算符 -bash: `\<' 附近有語法錯誤 [root@localhost ~]# [ 2 \< 3 ] && echo true || false true [root@localhost ~]# [ 2 < 3 ] && echo true || false -bash: 3: 沒有那個文件或目錄
3、大括號、花括號{}
3.1常規用法
- 花括號拓展。(通配(globbing))將對花括號中的文件名做擴展。在大括號中,不允許有空白,除非這個空白被引用或轉義。
- 對大括號中的以逗號分割的文件列表進行拓展。如 touch {a,b}.txt 結果為a.txt b.txt。
- 對大括號中以點點(..)分割的順序文件列表起拓展作用,如:touch {a..d}.txt 結果為a.txt b.txt c.txt d.txt。
- 代碼塊,又被稱為內部組,這個結構事實上創建了一個匿名函數 。與小括號中的命令不同,花括號內的命令不會新開一個子shell運行,即腳本余下部分仍可使用括號內變量。括號內的命令間用分號隔開,最后一個也必須有分號。{}的第一個命令和左括號之間必須要有一個空格。
3.2 幾種特殊的替換結構
- ${var:-string}:若變量var為空,則用在命令行中用string來替換${var:-string},否則變量var不為空時,則用變量var的值來替換${var:-string}。
[root@localhost ~]# a=2
[root@localhost ~]# echo $a
2
[root@localhost ~]# echo ${a:-123}
2
[root@localhost ~]# echo ${b:-123}
123
- ${var:=string}:替換規則同上,不同之處在於var為空時,string會賦給var。很常用的一種用法是,判斷某個變量是否賦值,沒有的話則給它賦上一個默認值。
[root@localhost ~]# echo ${b:=123}
123
[root@localhost ~]# echo $b
123
- ${var:+string}:替換規則和上面的相反,即只有當var不是空的時候才替換成string,若var為空時則不替換或者說是替換成變量 var的值,即空值。
[root@localhost ~]# echo ${b:+123}
123
[root@localhost ~]# echo ${c:+123}
- ${var:?string}替換規則為:若變量var不為空,則用變量var的值來替換${var:?string};若變量var為空,則把string輸出到標准錯誤中,並從腳本中退出。我們可利用此特性來檢查是否設置了變量的值。
tips:在上面替換結構中string不一定是常值的,可用另外一個變量的值或是一種命令的輸出。
3.3 四種模式匹配替換結構
# 是去掉左邊(在鍵盤上#在$之左邊) % 是去掉右邊(在鍵盤上%在$之右邊) #和%中的單一符號是最小匹配,兩個相同符號是最大匹配。
- ${variable%pattern}:shell在variable中查找,看它是否匹配給定的模式pattern結尾,如果是,就從命令行把variable中的內容去掉右邊最短的匹配模式。
- ${variable%%pattern}:shell在variable中查找,看它是否匹配給定的模式pattern結尾,如果是,就從命令行把variable中的內容去掉右邊最長的匹配模式。
- ${variable#pattern}:shell在variable中查找,看它是否匹配給定的模式pattern開始,如果是,就從命令行把variable中的內容去掉左邊最短的匹配模式。
- ${variable##pattern}:shell在variable中查找,看它是否匹配給定的模式pattern結尾,如果是,就從命令行把variable中的內容去掉右邊最長的匹配模式。
tips:這四種模式中都不會改變variable的值,其中,只有在pattern中使用了*匹配符號時,%和%%,#和##才有區別。結構中的pattern支持普通正則通配符,*表示零個或多個任意字符,?表示僅與一個任意字符匹配,[...]表示匹配中括號里面的字符,[!...]表示不匹配中括號里面的字符。
[root@localhost ~]# var=testcase [root@localhost ~]# echo $var testcase [root@localhost ~]# echo ${var%s*e} testca [root@localhost ~]# echo ${var%%s*e} te [root@localhost ~]# echo ${var#*s} tcase [root@localhost ~]# echo ${var##*s} e
3.4 字符串的提取和替換
- ${var:num}:shell在var中提取第num個字符到末尾的所有字符。若num為正數,從左邊0處開始;若num為負數,從右邊開始提取字串,但必須使用在冒號后面加空格或一個數字或整個num加上括號,如${var: -2}、${var:1-3}或${var:(-2)}。
- ${var:num1:num2}:num1是位置,num2是長度,表示從$var字符串的第$num1個位置開始提取長度為$num2的子串。不能為負數。
- ${var/pattern/pattern}:表示將var字符串的第一個匹配的pattern替換為另一個pattern。
- ${var//pattern/pattern}:表示將var字符串中的所有能匹配的pattern替換為另一個pattern。
[root@localhost ~]# var=/home/wwwroot/ [root@localhost ~]# echo $var /home/wwwroot/ [root@localhost ~]# echo ${var:5} /wwwroot/ [root@localhost ~]# echo ${var: -5} root/ [root@localhost ~]# echo ${var:1-5} oot/ [root@localhost ~]# echo ${var:1:4} home [root@localhost ~]# echo ${var/o/oo/} /hoo/me/wwwroot/ [root@localhost ~]# echo ${var/o/oo} /hoome/wwwroot/ [root@localhost ~]# echo $var /home/wwwroot/ [root@localhost ~]# echo ${var/o/oo} /hoome/wwwroot/ [root@localhost ~]# echo ${var//o/oo} /hoome/wwwroooot/
4、$符號后面的括號()
- ${a} 變量a的值, 在不引起歧義的情況下可以省略大括號。
- $(cmd) 命令替換,和
cmd
效果相同,結果為shell命令cmd的輸,不過某些Shell版本不支持$()形式的命令替換, 如tcsh。 - $((expression)) 和
exprexpression
效果相同, 計算數學表達式exp的數值。 其中exp只要符合C語言的運算規則即可, 甚至三目運算符和邏輯表達式都可以計算。
5、使用注意
- 單小括號,(cmd1;cmd2;cmd3) 新開一個子shell順序執行命令cmd1,cmd2,cmd3, 各命令之間用分號隔開, 最后一個命令后可以沒有分號。
- 單花括號,{ cmd1;cmd2;cmd3;} 在當前shell順序執行命令cmd1,cmd2,cmd3, 各命令之間用分號隔開, 最后一個命令后必須有分號, 第一條命令和左括號之間必須用空格隔開。
- 對{}和()而言, 括號中的重定向符只影響該條命令,而括號外的重定向符影響到括號中的所有命令。
作者:道無虛
鏈接:https://www.jianshu.com/p/b88c7e07aaa9
來源:簡書
著作權歸作者所有。商業轉載請聯系作者獲得授權,非商業轉載請注明出處。