Shell中() 、(())、[]、[[]]、{}的使用

() 的用法

子shell

()開啓子shell,外層不受影響正則表達式

#!/bin/bash
pwd
(cd /etc;pwd)
pwd
複製代碼

輸出:
/home/atvoid/文檔/個人堅果雲/project/ShellLife
/etc
/home/atvoid/文檔/個人堅果雲/project/ShellLifeshell

執行命令$()

等同於``,可是推薦使用$()數組

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 v2bash

[]的用法

內部符號 等同 test 不經常使用

test和[]中可用的比較運算符只有==和!=,二者都是用於字符串比較的,不可用於整數比較,整數比較只能使用-eq,-gt這種形式。不管是字符串比較仍是整數比較都不支持大於號小於號,若是實在想用,對於字符串比較能夠使用轉義形式,若是比較"ab"和"bc":[ ab \< bc ],結果爲真. [ ]中的邏輯與和邏輯或使用-a 和-o 表示 注意:ui

  1. 你必須在左括號的右側和右括號的左側各加一個空格,不然會報錯。
  2. test命令使用標準的數學比較符號來表示字符串的比較,而用文本符號來表示數值的比較。不少人會記反了。使用反了,shell可能得不到正確的結果。
  3. 大於符號或小於符號必需要轉義,不然會被理解成重定向。
#!/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]spa

(( ))及[[ ]] 經常使用

它們分別是[ ]的針對數學比較表達式和字符串表達式的增強版。 其中(( )),不須要再將表達式裏面的大小於符號轉義,除了能夠使用標準的數學運算符外,還增長了如下符號:命令行

在 [[ ]]中增長了另外一個特性:模式匹配

(()) 和 $(())

  1. 整數擴展.這種擴展計算是整數型的計算,不支持浮點型。((exp))結構擴展並計算一個算術表達式的值,若是表達式的結果爲0,那麼返回的退出狀態碼爲1,或者 是"假",而一個非零值的表達式所返回的退出狀態碼將爲0,或者是"true".如果邏輯判斷,表達式exp爲真則爲1,假則爲0.
  2. 只要括號中的運算符、表達式符合C語言運算規則,均可用在$((exp))中,甚至是三目運算符。做不一樣進位(如二進制、八進制、十六進制)運算時,輸出結果全都自動轉化成了十進制。如:echo $((16#5f)) 結果爲95 (16進位轉十進制)
  3. 單純用(( )) 也可重定義變量值,好比a=5; ((a++))可將 $a 重定義爲6
  4. 經常使用於算術運算比較,雙括號中的變量能夠不使用$符號前綴。括號內支持多個表達式用逗號分開。 只要括號中的表達式符合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
複製代碼

[[]]

  1. [[是 bash 程序語言的關鍵字。並非一個命令,[[ ]]結構比[ ]結構更加通用。在[[和]]之間全部的字符都不會發生文件名擴展或者單詞分割,可是會發生參數擴展和命令替換。
  2. 支持字符串的模式匹配,使用=~操做符時甚至支持shell的正則表達式。字符串比較時能夠把右邊的做爲一個模式,而不單單是一個字符串,比如[[ hello == hell? ]],結果爲真。[[ ]] 中匹配字符串或通配符,不須要引號。
  3. 使用[[ ... ]]條件判斷結構,而不是[ ... ],可以防止腳本中的許多邏輯錯誤。好比,&&、||、<和> 操做符可以正常存在於[[ ]]條件判斷結構中,可是若是出如今[ ]結構中的話,會報錯。好比能夠直接使用if [[ $a != 1 && $a != 2 ]], 若是不使用雙括號, 則爲if [ $a -ne 1] && [ $a != 2 ]或者if [ $a -ne 1 -a $a != 2 ]
  4. 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}code

    若變量var爲空,則用在命令行中用string來替換${var:-string},不然變量var不爲空時,則用變量var的值來替換${var:-string};對於${var:=string}的替換規則和${var:-string}是同樣的,所不一樣之處是${var:=string}若var爲空時,用string替換${var:=string}的同時,把string賦給變量var. ${var:=string}很經常使用的一種用法是,判斷某個變量是否賦值,沒有的話則給它賦上一個默認值。cdn

  • ${var:+string}blog

    與上面相反,即只有當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

相關文章
相關標籤/搜索