shell中的各種括弧的使用方法_linux shell

來源:互聯網
上載者:User

在這裡我想說的是幾種shell裡的小括弧,大括弧結構和有括弧的變數,命令的用法,如下:

1.${var}
2.$(cmd)
3.()和{}
4.${var:-string},${var:+string},${var:=string},${var:?string}
5.$((exp))
6.$(var%pattern),$(var%%pattern),$(var#pattern),$(var##pattern)

現在分述如下:


1.Shell中變數的原形:${var}
大家常見的變數形式都是$var,如

$ var=test
$ echo $var
test

但當你要顯示變數值加隨意的字元(我這裡用AA)時,就會出錯,如下:

$ echo $varAA

$

這時應該用變數的原形:${var},即是加一個大括弧來限定變數名稱的範圍,如下
$ echo ${var}AA
testAA
$

以這個特性,我們可以很方便地寫一個批量改尾碼名的程式,我把它命名為mymv,程式如下:
#!/bin/bash

tail=$1
for filename in `ls`
do
mv $filename ${filename}.$tail
done

程式需要提供一個尾碼名,如c,表示改為尾碼為c的C程式檔案,看下面的測試:
$ ls
a b c
$ mymv c
$ ls
a.c b.c c.c
$

看樣子程式啟動並執行很好,但這是不完善的程式,有2個要注意的問題:
A,目錄下沒有子目錄,如果有一個目錄,假設為dir,則也會被改為dir.c,這顯然不是我們想要的,應該修正這個程式能識別目錄。
B,沒有協助對程式的參數進行處理,程式應該足夠友好,在使用者沒有給定尾碼名時應能處理,像上面的將直接給檔案加上了一個點(.),這顯然也不是我們想要的。

因為我們的目的是說明${var},這樣已經足夠了,因此這裡不會再對上面的程式進行修正。

2.命令替換$(cmd)
命令替換$(cmd)和符號`cmd`(注意這不是單引號,在美式鍵盤上,`是ESC下面的那個鍵)有相同之處
$ ls
a b c
$ echo $(ls)
a b c
$ echo `ls`
a b c

我們來分析一下命令echo $(ls),以便理解所謂命令替換是什麼意思:
shell掃描一遍命令列,發現了$(cmd)結構,便將$(cmd)中的cmd執行一次,得到其標準輸出,再將此輸出放到原來命令echo $(ls)中的$(ls)位置,即替換了$(ls),再執行echo命令。
如下:
echo $(ls)被替換成了echo a b c
這裡要注意的是$(cmd)中的命令的錯誤輸出是不會被替換的,替換的只是標準輸出:
$ var=$(cat d) ###檔案d在目前的目錄不存在
cat: d: 沒有那個檔案或目錄
$ echo $var

$ ###顯然var變數的值是空的

3、一串的命令執行()和{}
()和{}都是對一串的命令進行執行,但有所區別:
A,()只是對一串命令重新開一個子shell進行執行
B,{}對一串命令在當前shell執行
C,()和{}都是把一串的命令放在括弧裡面,並且命令之間用;號隔開
D,()最後一個命令可以不用分號
E,{}最後一個命令要用分號
F,{}的第一個命令和左括弧之間必須要有一個空格
G,()裡的各命令不必和括弧有空格
H,()和{}中括弧裡面的某個命令的重新導向隻影響該命令,但括弧外的重新導向則影響到括弧裡的所有命令

我們來看幾個例子:
$ var=test
$ (var=notest; echo $var) ###變數var值為notest,此是在子shell中有效
notest
$ echo $var ###父shell中值仍為test
test
$ { var=notest; echo $var;} ###注意左括弧和var之間要有一個空格
notest
$ echo $var ###父shell中的var變數的值變為了notest
notest
$ { var1=test1;var2=test2;echo $var1>a;echo $var2;} ###輸出test1被重新導向到檔案a中,
test2 ###而test2輸出則仍輸出到標準輸出中。
$ cat a
test1
$ { var1=test1;var2=test2;echo $var1;echo $var2;}>a ###括弧內命令的標準輸出全部被重新導向到檔案a中
$ cat a
test1
test2

下面是一個腳步例子:

(
    echo "1"
    echo "2"
) | awk '{print NR,$0}'

4,幾種特殊的替換結構:${var:-string},${var:+string},${var:=string},${var:?string}
A,${var:-string}和${var:=string}
若變數var為空白,則用在命令列中用string來替換${var:-string},否則變數var不為空白時,則用變數var的值來替換${var:-string}
如:
$ echo $newvar

$ echo ${newvar:-a}
a
$ echo $newvar ###變數newvar的值仍然是空,但上一命令列中${newvar:-a}被替換成了a

$ newvar=b
$ echo ${newvar:-a} ###變數newvar的值不為空白時,此命令列中的${newvar:-b}被替換為$newvar,即b
b
$

對於${var:=string}的替換規則和${var:-string}是一樣的,所不同之處是${var:=string}若var為空白時,用string替換${var:=string}的同時,把string賦給變數var:


$ echo $newvar

$ echo ${newvar:=a}
a
$ echo $newvar ###變數newvar被賦值為a,同時${newvar:=a}被替換成a
a
$ echo ${newvar:=b} ###變數newvar不為空白(其值已被賦為a),則${newvar:=b}被替換為newvar的值(即b)
a
$ echo $newvar
a

${var:=string}很常用的一種用法是,判斷某個變數是否賦值,沒有的話則給它賦上一個預設值。
如設定預設的編輯器:
PHP 代碼:
echo You use editor: ${EDITOR:=/bin/vi}

B,${var:+string}
${var:+string}的替換規則和上面的相反,即只有當var不是空的時候才替換成string,若var為空白時則不替換或者說是替換成變數 var的值,即空值。(因為變數var此時為空白,所以這兩種說法是等價的)
$ echo $newvar
a
$ echo ${newvar:+b}
b
$ echo $newvar
a
$ newvar=
$ echo ${newvar:+b}

$

C,${var:?string}
替換規則為:若變數var不為空白,則用變數var的值來替換${var:?string};若變數var為空白,則把string輸出到標準錯誤中,並從指令碼中退出。我們可利用此特性來檢查是否設定了變數的值。
$ newvar=
$ echo ${newvar:?沒有設定newvar的值}
bash: newvar: 沒有設定newvar的值
$ newvar=a
$ echo ${newvar:?沒有設定newvar的值}
a
$

補充擴充:在上面這五種替換結構中string不一定是常值的,可用另外一個變數的值或是一種命令的輸出。
$ echo ${var:-`date`}
日 3月 6 02:10:39 CST 2005
$ echo ${var:-$(date)}
日 3月 6 02:11:46 CST 2005
$ a=test
$ echo ${var:-$a}
test
$

5.POSIX標準的擴充計算:$((exp))
這種計算是符合C語言的運算子,也就是說只要符合C的運算子都可用在$((exp)),甚至是三目運算子。
注意:這種擴充計算是整數型的計算,不支援浮點型.若是邏輯判斷,運算式exp為真則為1,假則為0。
$ echo $((3+2))
5
$ echo $((3>2))
1
$ echo $((25<3 ? 2:3))
3
$ echo $var

$ echo $((var=2+3))
5
$ echo $var
5
$ echo $((var++))
5
$ echo $var
6
$

好了,上面的例子足夠了,這也表明了這種擴充運算是很強大的。

6.四種模式比對替換結構:${var%pattern},${var%%pattern},${var#pattern},${var##pattern}
這四種結構的意義是:${var%pattern}和${var%%pattern}表示從最右邊(即結尾)匹配的,${var#pattern} 和${var##pattern}從最左邊(即開頭)匹配的。其中${var%pattern}和${var#pattern}是最短匹 配,${var%%pattern}和${var##pattern}是最長相符。只有在pattern中使用了萬用字元才能有最長最短的匹配,否則沒有最 長最短匹配之分。

結構中的pattern支援萬用字元,*表示零個或多個任一字元,?表示零個或一個任一字元,[...]表示匹配中括弧裡面的字元,[!...]表示不匹配中括弧裡面的字元。
$ var=aabbbccbbdbb
$ echo ${var%b}
aabbbccbbdb
$ echo ${var%%b}
aabbbccbbdb
$ echo ${var#a}
abbbccbbdbb
$ echo ${var##a}
abbbccbbdbb
$ echo ${var%*b}
aabbbccbbdb
$ echo ${var%%*b}

$ echo ${var#a*}
abbbccbbdbb
$ echo ${var##a*}

$

上面是簡單的例舉四種模式比對替換結構的用法。

•其他(見man bash中的Parameter Expansion)

${parameter/pattern/string}
              Pattern substitution.  The pattern is expanded to produce a pat‐
              tern  just  as in pathname expansion.  Parameter is expanded and
              the longest match of pattern against its value is replaced  with
              string.   If  pattern  begins with /, all matches of pattern are
              replaced  with  string.   Normally  only  the  first  match   is
              replaced.  If pattern begins with #, it must match at the begin‐
              ning of the expanded value of parameter.  If pattern begins with
              %,  it must match at the end of the expanded value of parameter.
              If string is null, matches of pattern are deleted and the / fol‐
              lowing pattern may be omitted.  If parameter is @ or *, the sub‐
              stitution operation is applied to each positional  parameter  in
              turn,  and the expansion is the resultant list.  If parameter is
              an array variable subscripted with  @  or  *,  the  substitution
              operation  is  applied  to each member of the array in turn, and
              the expansion is the resultant list.

(( )) :一對圓括弧有兩個地方用到。

1,for迴圈,

for (( expr1 ; expr2 ; expr3 ))

這裡一對雙括弧裡邊的運算式,GNU的文檔指出,expr1支援 Shell Arithmetic;expr2不為0時,expr3被賦值且語句執行。說的很麻煩,還要花時間搞清楚什麼是Shell Arithmetic。其實一言以蔽之,支援數字條件。比如:

for (( a=0 ; a<10 ; a++ )); do echo $a; done

會輸出 0 1 2 3 (帶換行哦~~~)

2,數學表達

(( )) 和 $(( ))

(( )) 的用法與let一樣,就不用多解釋了吧~~~

$(( ))就是把計算結果拿出來,可以用在雙引號裡邊,比如:

echo "1+2=$(( 1 + 2 ))"

會輸出 1+2=3

( ):一個圓括弧

在for迴圈裡,跟C文法一樣一樣的。

或者是子程式,返回整個裡邊表達的傳回值。裡邊的變數都是局部的,修改不會帶到外邊。舉例子

a=1

(a=3; echo $a)

echo a

結果是 3 1

還有個就是圈數組。。。這個就沒神馬意思了

[ ]:一個方括弧,是bash的命令,查man手冊是可以查到的,跟test一樣,在手冊裡可以看到很多用法。比如-b -c -gt -eq 什麼的很多,還有用-a表示與,-o表示或等等

[[ ]]:一對方括弧是一個方括弧的加強版,則是Shell的保留字,裡邊支援了 || && 等等這些符號。一般我喜歡用這個

還有相對複雜的 { }

幾個用處,區分變數,如:

var=abcd; echo ${var}EFG;

這樣,Bash就不會認為變數是varEFG了

還有用來截取字串的 ${ }文法比較靈活多變,這裡不做多解釋,大家有興趣可以自己去搜搜資料,一般我指令碼用到的字串處理,這個都能搞定了。

代碼塊。用來區分代碼的,但是跟( )有個區別,就是在末尾要加上 ;

1.()

在子shell中運行
(a=1);echo $a,結果是空,因為a=1不是在當前shell中啟動並執行(a=1);(echo $a)也是空的。不在同一個子shell中
數組的賦值,見最後的補充

2.(())

運算式計算
a=1;((a++)); echo $a,這時a就是2了。

3.<()和>()

進程代入,可以把命令的執行結果當成檔案一樣讀入
比如comm前一般需要sort,那就可以這樣comm <(sort 1.lst) <(sort 2.lst)
或者是paste <(cut -t2 file1) <(cut -t1 file1)
,和管道差不多,但是支援多個輸入。

4.$()

$(cmd) 執行cmd的結果,比如cmd是echo ls,那麼就是執行ls,比如file$(which bash),which bash的結果是/bin/bash,所以file $(which bash)等於file /bin/bash。如果你$(ls),而且你的目前的目錄下只有a b兩個檔案,那麼就是執行a b,然後系統會提示,命令沒找到。

5.$(())

運算式擴充,和(())很相似,但是這個是有點不同,$(())不能直接$((b++)),b=1;echo $((++b))這時b等於2,顯示的也是2,b=1;echo $((b++))這時b等於2,顯示的是1.

6.[]和[[]]

[]就是 test,[]和[[]]都是條件運算式,不過[[]]有比[]高的容錯性,如果a為空白,那麼[ $a -eq 0 ]會報錯,但是[[ $a -eq 0 ]]不會,所以一般都會使用[[]]或者是[ "$a" -eq 0 ],[[]]支援的功能也比[]多,比如[[ aaa =~a{3} ]],
[] 還有一種用途,如果你的目前的目錄下有a1-a9九個檔案,你可以用a[1-9]來替代這九個檔案。有點需要注意,你不能用a[1-20]來代替a1- a20,必須要a[1-9] a1[0-9] a20。

7.$[]

$(())的過去形式,現在已經不建議使用

8.{}

{1..30} 就是1-30,或者是/{,s}bin/表示/bin/和/sbin/,ab{c,d,e}表示abc、abd、abe

9.${}

變數,用法很多,可以查看man bash。
先寫這些,以後想到再補充吧。

補充:()同時也是數組的賦值,比如a=(1 3 5),那麼${a[0]}=1;${a[1]}=3;${a[2]}=5,需要注意的是,下標是從0開始的

Shell中的括弧有其特殊的用法, 現總結如下:

1. 符號$後的括弧

${a} 變數a的值, 在不引起歧義的情況下可以省略大括弧.
$(cmd) 命令替換, 結果為shell命令cmd的輸出, 和`cmd`效果相同, 不過某些Shell版本不支援$()形式的命令替換, 如tcsh.
$((exp)) 和`expr exp`效果相同, 計算數學運算式exp的數值, 其中exp只要符合C語言的運算規則即可, 甚至三目運算子和邏輯運算式都可以計算.

2. 多條命令執行

(cmd1;cmd2;cmd3) 新開一個子shell順序執行命令cmd1,cmd2,cmd3, 各命令之間用分號隔開, 最後一個命令後可以沒有分號.
{ cmd1;cmd2;cmd3;} 在當前shell順序執行命令cmd1,cmd2,cmd3, 各命令之間用分號隔開, 最後一個命令後必須有分號, 第一條命令和左括弧之間必須用空格隔開.
對{}和()而言, 括弧中的重新導向符隻影響該條命令, 而括弧外的重新導向符影響到括弧中的所有命令.

3. 雙括弧的特殊用法

(()) 增強括弧的用法, 常用於算術運算比較. 雙括弧中的變數可以不使用$符號首碼, 只要括弧中的運算式符合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 ].
[[]] 增強方括弧用法, 常用於字串的比較. 主要用於條件測試, 雙括弧中的運算式可以使用&&, ||, <, >等C語言文法.
比如可以直接使用if [[ $a != 1 && $a != 2 ]], 如果不適用雙括弧, 則為if [ $a -ne 1] && [ $a != 2 ]或者if [ $a -ne 1 -a $a != 2 ].

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.