Shell中() 、(())、[]、[[]]、{}的使用
() 的用法
子shell
()開啟子shell,外層不受影響
#!/bin/bash
pwd
(cd /etc;pwd)
pwd
複製程式碼
輸出:
/home/atvoid/檔案/我的堅果雲/project/ShellLife
/etc
/home/atvoid/檔案/我的堅果雲/project/ShellLife
執行命令$()
等同於``,但是推薦使用$()
a=$(echo 123)
echo $a #輸出123
複製程式碼
初始化陣列/字典
#!/bin/bash
array=(a b c d)
echo ${array[*]}
declare -A dict=([k1]="v1" [k2]="v2")
echo ${dict[*]}
複製程式碼
輸出
a b c d
v1 v2
[]的用法
內部符號 等同 test 不常用
test和[]中可用的比較運運算元只有==和!=,兩者都是用於字串比較的,不可用於整數比較,整數比較只能使用-eq,-gt這種形式。無論是字串比較還是整數比較都不支援大於號小於號,如果實在想用,對於字串比較可以使用轉義形式,如果比較"ab"和"bc":[ ab \< bc ]
,結果為真.
[ ]中的邏輯與和邏輯或使用-a 和-o 表示
注意:
- 你必須在左括號的右側和右括號的左側各加一個空格,否則會報錯。
- test命令使用標準的數學比較符號來表示字串的比較,而用文字元號來表示數值的比較。很多人會記反了。使用反了,shell可能得不到正確的結果。
- 大於符號或小於符號必須要轉義,否則會被理解成重定向。
#!/bin/bash
a=1
b=2
if [ $a -lt $b ]; then
echo a小於b
fi
a="a"
b="b"
if [ $a != $b ]; then
echo a不等於b
fi
複製程式碼
其他
引用陣列元素 array[1]
正則 [0-9]
(( ))及[[ ]] 常用
它們分別是[ ]的針對數學比較表示式和字串表示式的加強版。 其中(( )),不需要再將表示式裡面的大小於符號轉義,除了可以使用標準的數學運運算元外,還增加了以下符號:
在 [[ ]]中增加了另一個特性:模式匹配(()) 和 $(())
- 整數擴充套件.這種擴充套件計算是整數型的計算,不支援浮點型。
((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 ]
。
#!/bin/bash
#重新定義變數
a=5
((a++))
echo $a # 輸出6
#for迴圈
for ((i = 0; i < 5; i++)); do
echo 當前序號: $i
done
#條件判斷
if ((1 + 2)); then
echo 真
fi
if !((1 - 1)); then
echo 假
fi
#算數運算
echo $((1 + 2 * 3)) # 輸出7
複製程式碼
[[]]
-
[[
是 bash 程式語言的關鍵字。並不是一個命令,[[ ]]
結構比[ ]
結構更加通用。在[[和]]之間所有的字元都不會發生檔名擴充套件或者單詞分割,但是會發生引數擴充套件和命令替換。 - 支援字串的模式匹配,使用=~操作符時甚至支援shell的正則表示式。字串比較時可以把右邊的作為一個模式,而不僅僅是一個字串,比
如[[ hello == hell? ]]
,結果為真。[[ ]]
中匹配字串或萬用字元,不需要引號。 - 使用
[[ ... ]]
條件判斷結構,而不是[ ... ]
,能夠防止指令碼中的許多邏輯錯誤。比如,&&、||、<和> 操作符能夠正常存在於[[ ]]
條件判斷結構中,但是如果出現在[ ]
結構中的話,會報錯。比如可以直接使用if [[ $a != 1 && $a != 2 ]]
,則為if [ $a -ne 1] && [ $a != 2 ]
或者if [ $a -ne 1 -a $a != 2 ]
。 - bash把雙中括號中的表示式看作一個單獨的元素,並返回一個退出狀態碼。
#!/bin/bash
if [[ hello == hell? ]];then
echo 'hello == hell? true'
fi
if [[ 5 > 4 ]];then
echo '5 > 4 true'
fi
if [[ 123 =~ ^[0-9]+ ]];then
echo '123 match ^[0-9]+'
fi
複製程式碼
{}的用法
變數替換${}
#!/bin/bash
array=(a b c d)
# echo $array[*] 會輸出a[*]
echo ${array[*]} # 可以輸出a b c d
複製程式碼
特殊替換
-
${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輸出到標準錯誤中,並從指令碼中退出。我們可利用此特性來檢查是否設定了變數的值。
字串提取和替換
-
${var:num}
從num開始到結尾,0計數
-
${var:num1:num2}
從num1開始,提取num2個,0計數
-
${var/pattern/pattern}
替換第一個
-
${var//pattern/pattern}
全部替換
var="01234560"
echo $var #01234560
echo ${var:5} #560
echo ${var: -6} #234560
echo ${var:(-6)} #234560
echo ${var:1:4} #1234
echo ${var/0/-1} #-11234560
echo ${var//0/-1} #-1123456-1
複製程式碼
四種模式匹配替換結構
#
是去掉左邊(在鍵盤上#在$之左邊)
%
是去掉右邊(在鍵盤上%在$之右邊)
#
和%
中的單一符號是最小匹配,兩個相同符號是最大匹配。
-
${var#pattern}
${variable#pattern} 這種模式時,shell在variable中查詢,看它是否一給的模式pattern開始,如果是,就從命令列把variable中的內容去掉左邊最短的匹配模式
-
${var##pattern}
${variable##pattern} 這種模式時,shell在variable中查詢,看它是否一給的模式pattern結尾,如果是,就從命令列把variable中的內容去掉右邊最長的匹配模式
-
${var%pattern}
${variable%pattern},這種模式時,shell在variable中查詢,看它是否一給的模式pattern結尾,如果是,就從命令列把variable中的內容去掉右邊最短的匹配模式
-
${var%%pattern}
${variable%%pattern},這種模式時,shell在variable中查詢,看它是否一給的模式pattern結尾,如果是,就從命令列把variable中的內容去掉右邊最長的匹配模式
var=testcase
echo $var #testcase
echo ${var%s*e} #testca
echo ${var%%s*e} #te
echo ${var#?e} #stcase
echo ${var##?e}#stcase
echo ${var##*e} #空
echo ${var##*s} #e
echo ${var##test} #case
複製程式碼
表示範圍
touch {a..d}.txt
結果為a.txt b.txt c.txt d.txt