【shell】shell中各類括號的做用()、(())、[]、[[]]、{}

1、小括號,圓括號()

一、單小括號 ()

   ①命令組。括號中的命令將會新開一個子shell順序執行,因此括號中的變量不可以被腳本餘下的部分使用。括號中多個命令之間用分號隔開,最後一個命令能夠沒有分號,各命令和括號之間沒必要有空格。
   ②命令替換。等同於`cmd`,shell掃描一遍命令行,發現了$(cmd)結構,便將$(cmd)中的cmd執行一次,獲得其標準輸出,再將此輸出放到原來命令。有些shell不支持,如tcsh。
    ③用於初始化數組。如:array=(a b c d)
 
 

二、雙小括號 (( ))

    ①整數擴展。這種擴展計算是整數型的計算,不支持浮點型。((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 ]。

2、中括號,方括號[]

  一、單中括號 []

    ①bash 的內部命令,[和test是等同的。若是咱們不用絕對路徑指明,一般咱們用的都是bash自帶的命令。if/test結構中的左中括號是調用test的命令標識,右中括號是關閉條件判斷的。這個命令把它的參數做爲比較表達式或者做爲文件測試,而且根據比較的結果來返回一個退出狀態碼。if/test結構中並非必須右中括號,可是新版的Bash中要求必須這樣。
    ②Test和[]中可用的比較運算符只有=和!=,二者都是用於字符串比較的,不可用於整數比較,整數比較只能使用-eq,-gt這種形式。不管是字符串比較仍是整數比較都不支持大於號小於號。若是實在想用,對於字符串比較可使用轉義形式,若是比較"ab"和"bc":[ ab \< bc ],結果爲真,也就是返回狀態爲0。[ ]中的邏輯與和邏輯或使用-a 和-o 表示。且[]先後都有空格。
    ③字符範圍。用做正則表達式的一部分,描述一個匹配的字符範圍。做爲test用途的中括號內不能使用正則。
    ④在一個array 結構的上下文中,中括號用來引用數組中每一個元素的編號。
 
 

 二、雙中括號[[ ]]

    ①[[是 bash 程序語言的關鍵字。並非一個命令,[[ ]] 結構比[ ]結構更加通用。在[[和]]之間全部的字符都不會發生文件名擴展或者單詞分割,可是會發生參數擴展和命令替換。
    ②支持字符串的模式匹配,使用=~操做符時甚至支持shell的正則表達式。字符串比較時能夠把右邊的做爲一個模式,而不只僅是一個字符串,好比[[ hello == hell? ]],結果爲真。[[ ]] 中匹配字符串或通配符,不須要引號。
    ③使用[[ ... ]]條件判斷結構,而不是[ ... ],可以防止腳本中的許多邏輯錯誤。好比,&&、||、<和> 操做符可以正常存在於[[ ]]條件判斷結構中,可是若是出如今[ ]結構中的話,會報錯。好比能夠直接使用if [[ $a != 1 && $a != 2 ]], 若是不適用雙括號, 則爲if [ $a -ne 1] && [ $a != 2 ]或者if [ $a -ne 1 -a $a != 2 ]。
    ④bash把雙中括號中的表達式看做一個單獨的元素,並返回一個退出狀態碼。
例子:
[cpp]  view plain  copy
 
  1. if ($i<5)    
  2. if [ $i -lt 5 ]    
  3. if [ $a -ne 1 -a $a != 2 ]    
  4. if [ $a -ne 1] && [ $a != 2 ]    
  5. if [[ $a != 1 && $a != 2 ]]    
  6.      
  7. for i in $(seq 0 4);do echo $i;done    
  8. for i in `seq 0 4`;do echo $i;done    
  9. for ((i=0;i<5;i++));do echo $i;done    
  10. for i in {0..4};do echo $i;done    

3、大括號、花括號 {}

一、常規用法

    ①大括號拓展。(通配(globbing))將對大括號中的文件名作擴展。在大括號中,不容許有空白,除非這個空白被引用或轉義。第一種:對大括號中的以逗號分割的文件列表進行拓展。如 touch {a,b}.txt 結果爲a.txt b.txt。第二種:對大括號中以點點(..)分割的順序文件列表起拓展做用,如:touch {a..d}.txt 結果爲a.txt b.txt c.txt d.txt
例如:
 
複製代碼
[root@iz2ze46xi6pjjj69ailg9lz ttt]# touch test{0..5}.txt
[root@iz2ze46xi6pjjj69ailg9lz ttt]# ls
test0.txt  test1.txt  test2.txt  test3.txt  test4.txt  test5.txt
[root@iz2ze46xi6pjjj69ailg9lz ttt]# echo str{b..d}
strb strc strd
[root@iz2ze46xi6pjjj69ailg9lz ttt]# echo str{b,d}
strb strd
複製代碼

 

 

    ②代碼塊,又被稱爲內部組,這個結構事實上建立了一個匿名函數 。與小括號中的命令不一樣,大括號內的命令不會新開一個子shell運行,即腳本餘下部分仍可以使用括號內變量。括號內的命令間用分號隔開,最後一個也必須有分號。{}的第一個命令和左括號之間必需要有一個空格。{}也能夠用於多行註釋,做爲函數包起來只是不調用便可。
 
 

  二、幾種特殊的替換結構

${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不必定是常值的,可用另一個變量的值或是一種命令的輸出。

 三、四種模式匹配替換結構

模式匹配記憶方法:
# 是去掉左邊(在鍵盤上#在$之左邊)
% 是去掉右邊(在鍵盤上%在$之右邊)
#和%中的單一符號是最小匹配,兩個相同符號是最大匹配。html

${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支持通配符,*表示零個或多個任意字符,?表示僅與一個任意字符匹配,[...]表示匹配中括號裏面的字符,[!...]表示不匹配中括號裏面的字符。
[cpp]  view plain  copy
 
  1. # var=testcase    
  2. # echo $var    
  3. testcase    
  4. # echo ${var%s*e}   
  5. testca    
  6. # echo $var    
  7. testcase   
  8. # echo ${var%%s*e}   
  9. te  
  10. # echo ${var#?e}    
  11. stcase  
  12. # echo ${var##?e}    
  13. stcase  
  14. # echo ${var##*e}    
  15.   
  16. # echo ${var##*s}    
  17. e    
  18. # echo ${var##test}    
  19. case    

 

 四、字符串提取和替換

${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。
[html]  view plain  copy
 
  1. [root@centos ~]# var=/home/centos  
  2. [root@centos ~]# echo $var  
  3. /home/centos  
  4. [root@centos ~]# echo ${var:5}  
  5. /centos  
  6. [root@centos ~]# echo ${var: -6}  
  7. centos  
  8. [root@centos ~]# echo ${var:(-6)}  
  9. centos  
  10. [root@centos ~]# echo ${var:1:4}  
  11. home  
  12. [root@centos ~]# echo ${var/o/h}  
  13. /hhme/centos  
  14. [root@centos ~]# echo ${var//o/h}  
  15. /hhme/cenths  

 

4、符號$後的括號

(1)${a} 變量a的值, 在不引發歧義的狀況下能夠省略大括號。python

(2)$(cmd) 命令替換,和`cmd`效果相同,結果爲shell命令cmd的輸,過某些Shell版本不支持$()形式的命令替換, 如tcsh。正則表達式

(3)$((expression)) 和`exprexpression`效果相同, 計算數學表達式exp的數值, 其中exp只要符合C語言的運算規則便可, 甚至三目運算符和邏輯表達式均可以計算。shell

5、使用

一、多條命令執行express

(1)單小括號,(cmd1;cmd2;cmd3) 新開一個子shell順序執行命令cmd1,cmd2,cmd3, 各命令之間用分號隔開, 最後一個命令後能夠沒有分號。centos

(2)單大括號,{ cmd1;cmd2;cmd3;} 在當前shell順序執行命令cmd1,cmd2,cmd3, 各命令之間用分號隔開, 最後一個命令後必須有分號, 第一條命令和左括號之間必須用空格隔開。
對{}和()而言, 括號中的重定向符隻影響該條命令, 而括號外的重定向符影響到括號中的全部命令。數組

 

 

 

 

 

例如:bash

複製代碼
#!/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
複製代碼

 

 

運行結果:less

複製代碼
#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

 

 

 

 更多的特殊符號參考:http://www.jb51.net/article/69966.htm

相關文章
相關標籤/搜索