一、小括號,圓括號()

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
④常用于算術運算比較,雙括號中的變數可以不使用符號前綴,括號內支持多個運算式用逗號分開,只要括號中的運算式符合C語言運算規則,比如可以直接使用for((i=0;i<5;i++)),如果不使用雙括號,則為foriin‘seq04‘或者foriin0..4,再如可以直接使用if((i<5)), 如果不使用雙括號, 則為if [ $i -lt 5 ],
二、中括號,方括號[]
1、單中括號 []
①bash 的內部命令,[和test是等同的,如果我們不用絕對路徑指明,通常我們用的都是bash自帶的命令,if/test結構中的左中括號是呼叫test的命令標識,右中括號是關閉條件判斷的,這個命令把它的引數作為比較運算式或者作為檔案測驗,并且根據比較的結果來回傳一個退出狀態碼,if/test結構中并不是必須右中括號,但是新版的Bash中要求必須這樣,
②Test和[]中可用的比較運算子只有==和!=,兩者都是用于字串比較的,不可用于整數比較,整數比較只能使用-eq,-gt這種形式,無論是字串比較還是整數比較都不支持大于號小于號,如果實在想用,對于字串比較可以使用轉義形式,如果比較"ab"和"bc":[ ab < bc ],結果為真,也就是回傳狀態為0,[ ]中的邏輯與和邏輯或使用-a 和-o 表示,
③字符范圍,用作正則運算式的一部分,描述一個匹配的字符范圍,作為test用途的中括號內不能使用正則,
④在一個array 結構的背景關系中,中括號用來參考陣列中每個元素的編號,
2、雙中括號[[ ]]
①[[是 bash 程式語言的關鍵字,并不是一個命令,[[ ]] 結構比[ ]結構更加通用,在[[和]]之間所有的字符都不會發生檔案名擴展或者單詞分割,但是會發生引數擴展和命令替換,
②支持字串的模式匹配,使用=~運算子時甚至支持shell的正則運算式,字串比較時可以把右邊的作為一個模式,而不僅僅是一個字串,比如[[ hello == hell? ]],結果為真,[[ ]] 中匹配字串或通配符,不需要引號,
③使用[[ ... ]]條件判斷結構,而不是[ ... ],能夠防止腳本中的許多邏輯錯誤,比如,&&、||、<和> 運算子能夠正常存在于[[ ]]條件判斷結構中,但是如果出現在[ ]結構中的話,會報錯,比如可以直接使用if [[ $a != 1 && $a != 2 ]], 如果不適用雙括號, 則為if [ $a -ne 1] && [ $a != 2 ]或者if [ $a -ne 1 -a $a != 2 ],
④bash把雙中括號中的運算式看作一個單獨的元素,并回傳一個退出狀態碼,
例子:
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、常規用法
①大括號拓展,(通配(globbing))將對大括號中的檔案名做擴展,在大括號中,不允許有空白,除非這個空白被參考或轉義,第一種:對大括號中的以逗號分割的檔案串列進行拓展,如 touch {a,b}.txt 結果為a.txt b.txt,第二種:對大括號中以點點(..)分割的順序檔案串列起拓展作用,如:touch {a..d}.txt 結果為a.txt b.txt c.txt d.txt
# 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
②代碼塊,又被稱為內部組,這個結構事實上創建了一個匿名函式 ,與小括號中的命令不同,大括號內的命令不會新開一個子shell運行,即腳本余下部分仍可使用括號內變數,括號內的命令間用分號隔開,最后一個也必須有分號,{}的第一個命令和左括號之間必須要有一個空格,
2、幾種特殊的替換結構
${var:-string},${var:+string},${var:=string},${var:?string}
①${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}很常用的一種用法是,判斷某個變數是否賦值,沒有的話則給它賦上一個默認值,
② ${var:+string}的替換規則和上面的相反,即只有當var不是空的時候才替換成string,若var為空時則不替換或者說是替換成變數 var的值,即空值,(因為變數var此時為空,所以這兩種說法是等價的)
③${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支持通配符,表示零個或多個任意字符,?表示僅與一個任意字符匹配,[...]表示匹配中括號里面的字符,[!...]表示不匹配中括號里面的字符,
# 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
4、字串提取和替換
${var:num},${var:num1:num2},${var/pattern/pattern},${var//pattern/pattern}
第一種模式:${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@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
四、符號$后的括號
(1)${a} 變數a的值, 在不引起歧義的情況下可以省略大括號,
(2)$(cmd) 命令替換,和cmd效果相同,結果為shell命令cmd的輸,過某些Shell版本不支持$()形式的命令替換, 如tcsh,
(3)$((expression)) 和exprexpression效果相同, 計算數學運算式exp的數值, 其中exp只要符合C語言的運算規則即可, 甚至三目運算子和邏輯運算式都可以計算,
五、使用
多條命令執行
(1)單小括號,(cmd1;cmd2;cmd3) 新開一個子shell順序執行命令cmd1,cmd2,cmd3, 各命令之間用分號隔開, 最后一個命令后可以沒有分號,
(2)單大括號,{ cmd1;cmd2;cmd3;} 在當前shell順序執行命令cmd1,cmd2,cmd3, 各命令之間用分號隔開, 最后一個命令后必須有分號, 第一條命令和左括號之間必須用空格隔開,
對{}和()而言, 括號中的重定向符只影響該條命令, 而括號外的重定向符影響到括號中的所有命令,
以上就是良許教程網為各位朋友分享的Linux Shell 中 ()、(())、[]、[[]]、{} 的作用,
本文由博客一文多發平臺 OpenWrite 發布!
轉載請註明出處,本文鏈接:https://www.uj5u.com/caozuo/254256.html
標籤:Linux
上一篇:Linux 修改 ELF 解決 glibc 兼容性問題
下一篇:Linux初學者的七個建議
