Linux--shell--3--()、(())、[]、[[]]、{}
技術標籤:shell
一、小括號,圓括號()
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++)), 如果不使用雙括號, 則為for i in seq 0 4
二、中括號,方括號[]
1、單中括號 []
① bash 的內部命令,[和test是等同的。如果我們不用絕對路徑指明,通常我們用的都是bash自帶的命令。if/test結構中的左中括號是呼叫test的命令標識,右中括號是關閉條件判斷的。這個命令把它的引數作為比較表示式或者作為檔案測試,並且根據比較的結果來返回一個退出狀態碼。if/test結構中並不是必須右中括號,但是新版的Bash中要求必須這樣。
② Test和[]中可用的比較運算子只有=和!=,兩者都是用於字串比較的,不可用於整數比較,整數比較只能使用-eq,-gt這種形式
[]前後都有空格
。
③ 字元範圍。用作正則表示式的一部分,描述一個匹配的字元範圍。作為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;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
三、大括號、花括號 {}
1、常規用法
① 大括號拓展。(通配(globbing))將對大括號中的檔名做擴充套件。在大括號中,不允許有空白,除非這個空白被引用或轉義。第一種:對大括號中的以逗號分割的檔案列表進行拓展。如 touch {a,b}.txt 結果為a.txt b.txt。第二種:對大括號中以點點(…)分割的順序檔案列表起拓展作用,如:touch {a…d}.txt 結果為a.txt b.txt c.txt d.txt
例如:
[[email protected] ttt]# touch test{0..5}.txt
[[email protected] ttt]# ls
test0.txt test1.txt test2.txt test3.txt test4.txt test5.txt
[[email protected] ttt]# echo str{b..d}
strb strc strd
[[email protected] ttt]# echo str{b,d}
strb strd
②程式碼塊,又被稱為內部組,這個結構事實上建立了一個匿名函式 。與小括號中的命令不同,大括號內的命令不會新開一個子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加上括號,如
v
a
r
:
−
2
、
{var: -2}、
var:−2、{var:1-3}或${var:(-2)}。
第二種模式:${var:num1:num2},num1是位置,num2是長度。表示從 v a r 字 符 串 的 第 var字串的第 var字符串的第num1個位置開始提取長度為$num2的子串。不能為負數。
第三種模式:${var/pattern/pattern}表示將var字串的第一個匹配的pattern替換為另一個pattern。
第四種模式:${var//pattern/pattern}表示將var字串中的所有能匹配的pattern替換為另一個pattern。
[[email protected] ~]# var=/home/centos
[[email protected] ~]# echo $var
/home/centos
[[email protected] ~]# echo ${var:5}
/centos
[[email protected] ~]# echo ${var: -6}
centos
[[email protected] ~]# echo ${var:(-6)}
centos
[[email protected] ~]# echo ${var:1:4}
home
[[email protected] ~]# echo ${var/o/h}
/hhme/centos
[[email protected] ~]# echo ${var//o/h}
/hhme/cenths
四、符號$後的括號
(1)${a} 變數a的值, 在不引起歧義的情況下可以省略大括號。
(2)$(cmd) 命令替換,和cmd
效果相同,結果為shell命令cmd的輸,過某些Shell版本不支援$()形式的命令替換, 如tcsh。
(3)$((expression)) 和exprexpression
效果相同, 計算數學表示式exp的數值, 其中exp只要符合C語言的運算規則即可, 甚至三目運算子和邏輯表示式都可以計算。
五、使用
1、多條命令執行
(1)單小括號,(cmd1;cmd2;cmd3) 新開一個子shell順序執行命令cmd1,cmd2,cmd3, 各命令之間用分號隔開, 最後一個命令後可以沒有分號。
(2)單大括號,{ cmd1;cmd2;cmd3;} 在當前shell順序執行命令cmd1,cmd2,cmd3, 各命令之間用分號隔開, 最後一個命令後必須有分號, 第一條命令和左括號之間必須用空格隔開。
對{}和()而言, 括號中的重定向符隻影響該條命令, 而括號外的重定向符影響到括號中的所有命令。
例如:
#!/bin/bash
str1="test1"
str2="Test1"
num1=33
num2=4
###################string complare#########
#1 [] use =
if [ "$str1" = "$str2" ]
then
echo "#1 ${str1} equals to ${str2}"
else
echo "#1 ${str1} not e2 $str2"
fi
#2 [] use !=
if [ "$str1" != "$str2" ]
then
echo "#2 ${str1} not equals to ${str2}"
else
echo "#2 ${str1} eq2 $str2"
fi
#3 [] use \< (注意字串之間不能有空格)
if [ "${str1}"\<"${str2}" ]
then
echo "#3 ${str1} less than ${str2}"
else
echo "#3 ${str1} eq2 or grater than $str2"
fi
#4 [[]] use <
if [[ "$str1" < "$str2" ]]
then
echo "#4 ${str1} less than ${str2}"
else
echo "#4 ${str1} eq2 or grater than $str2"
fi
#################### number complare ############################
#1 [] use -lt.-gt,-ge
if [ "$num1" -lt "${num2}" ]
then
echo "${num1} less than ${num2}"
else
echo "${num1} eq2 or grater than $num2"
fi
#2 [] use \< is complare as string (錯誤的示範,不能在[]中使用轉義比較數字,會當成字串比較)
if [ "$num1" \< "${num2}" ]
then
echo "${num1} less than ${num2}"
else
echo "${num1} eq2 or grater than $num2"
fi
#3 (()) use <
if (( "$num1" < "${num2}" ))
then
echo "${num1} less than ${num2}"
else
echo "${num1} eq2 or grater than $num2"
fi
#####################(()) use to number +-*/ ###########################
echo "############# (()) use ro number +-*/% ##############"
echo $(($num1 + $num2))
echo $(($num1 - $num2))
echo $(($num1 * $num2))
echo $(($num1 / $num2))
echo $(($num1 % $num2))
####################### () ###################
echo "############# \$() use like \`\` ##################"
echo `which pwd`
echo $(which pwd)
##################### \${} to get variables #########################
echo "################## \${var} is like \$var ################"
echo ${str1}
echo $str1
執行結果:
#1 test1 not e2 Test1
#2 test1 not equals to Test1
#3 test1 less than Test1
#4 test1 less than Test1
33 eq2 or grater than 4
33 less than 4
33 eq2 or grater than 4
############# (()) use ro number +-*/% ##############
37
29
132
8
1
############# $() use like `` ##################
/usr/bin/pwd
/usr/bin/pwd
################## ${var} is like $var ################
test1
test1
總結:
(1) $(cmd)與··(鍵盤上1左邊的~)一樣,都是命令替換,可以將執行結果提取出來
(2) []使用的時候[ ]前後都必須有空格,且兩個字元或數字之間的比較符左右也必須有空格。
(3) []是test的另一種形式,[]中間只能使用= 和 != 比較字串,如果使用< 、>需要進行轉義\。
[]中間如果比較數字需要用 -lt 等符號,不能使用<比較數字,會當成字串處理。
(4) [[]]可用於處理邏輯命令,也可以用於處理字串是否相等,且使用<、>不用轉義符.
(5) (())可用於比較數字,且不用轉義,而且也可以用於數字計算,比較的時候也是用普通的>,<。(())計算的時候運算子與數字之間不能有空格,例如: sum=$(($sum+4))
(6) 字串比較 用[],與普通的<,>,=,!=符號,如果使用<,>需要轉義;或者使用[[]]比較字串也是用普通符號不用轉義
數字比較用[]的時候用-lt,-gt等符號,不能使用<(因為會當成字串處理);或者用(())比較數字用普通符號不用轉義
(7) 可以將$理解為取變數的符號,$var 或者 ${} ,在不影響語義的情況下可以省去{},但是最好寫上{}
。例如:test=XXX.$testWWWW.這時候就必須加上{}變為${test}WWWW