歡迎來到Linux教程網
Linux教程網
Linux教程網
Linux教程網
Linux教程網 >> Linux編程 >> SHELL編程 >> Linux下高效編寫Shell——Shell特殊字符匯總

Linux下高效編寫Shell——Shell特殊字符匯總

日期:2017/2/28 13:59:20   编辑:SHELL編程

Linux下無論如何都是要用到shell命令的,在Shell的實際使用中,有編程經驗的很容易上手,但稍微有難度的是shell裡面的那些個符號,各種特殊的符號在我們編寫Shell腳本的時候如果能夠用的好,往往能給我們起到事半功倍的效果,為此,特地將Shell裡面的一些符號說明羅列成對照表的形式,以便快速的查找。看看你知道下表中哦你的哪些Shell符號呢?

Shell符號及各種解釋對照表:

Shell符號使用方法及說明 #

注釋符號(Hashmark[Comments])

1.在shell文件的行首,作為shebang標記,#!/bin/bash;

2. 其他地方作為注釋使用,在一行中,#後面的內容並不會被執行,除非;

3. 但是用單/雙引號包圍時,#作為#號字符本身,不具有注釋作用。

;

作為多語句的分隔符(Command separator [semicolon])。

多個語句要放在同一行的時候,可以使用分號分隔。注意,有時候分號需要轉義。

;;

連續分號(Terminator [double semicolon])。

在使用case選項的時候,作為每個選項的終結符。在Bash version 4+ 的時候,還可以使用[;;&], [;&]

.

點號(dot command [period])。

1. 相當於bash內建命令source,如:

  1. #!/bin/bash
  2. . data-file
  3. #包含data-file;

2. 作為文件名的一部分,在文件名的開頭,表示該文件為隱藏文件,ls一般不顯示出來(ls -a 可以顯示);

3. 作為目錄名,一個點代表當前目錄,兩個點號代表上層目錄(當前目錄的父目錄)。注意,兩個以上的點不出現,除非你用引號(單/雙)包圍作為點號字符本身;

4. 正則表達式中,點號表示任意一個字符。

"

雙引號(partial quoting [double quote])。

部分引用。雙引號包圍的內容可以允許變量擴展,也允許轉義字符的存在。如果字符串內出現雙引號本身,需要轉義,因此不一定雙引號是成對的。

'

單引號(full quoting [single quote])。

單引號括住的內容,被視為單一字符串,引號內的禁止變量擴展,所有字符均作為字符本身處理(除單引號本身之外),單引號必須成對出現。

,

逗號(comma operator [comma])。

1. 用在連接一連串的數學表達式中,這串數學表達式均被求值,但只有最後一個求值結果被返回。如:

  1. #!/bin/bash
  2. let t1=((a=5+1, b=7+2))
  3. echo t1=$t1, a=$a, b=$b
  4. ## 這個$t1=$b;

2. 用於參數替代中,表示首字母小寫,如果是兩個逗號,則表示全部小寫,注意,這個特性在bash version 4的時候被添加的。例子:

  1. a="ATest"
  2. echo ${a,}
  3. echo ${a,,}
  4. ## 前面輸出aTest,後面輸出的是atest。
\

反斜線,反斜桿(escape [backslash])。

1. 放在特殊符號之前,轉義特殊符號的作用,僅表示特殊符號本身,這在字符串中常用;

2. 放在一行指令的最末端,表示緊接著的回車無效(其實也就是轉義了Enter),後繼新行的輸入仍然作為當前指令的一部分。

/

斜線,斜桿(Filename path separator [forward slash])。

1.作為路徑的分隔符,路徑中僅有一個斜桿表示根目錄,以斜桿開頭的路徑表示從根目錄開始的路徑;

2.在作為運算符的時候,表示除法符號。如:a=4/2

`

反引號,後引號(Command substitution[backquotes])。

命令替換。這個引號包圍的為命令,可以執行包圍的命令,並將執行的結果賦值給變量。如:a=`dirname '/tmp/x.log'` 。後面dirname返回的結果會賦值給a,注意,此處Mitchell特地使用了反引號和單引號,注意區別。

:

冒號(null command [colon])。

空命令,這個命令什麼都不做,但是有返回值,返回值為0(即:true)。這個命令的作用非常奇妙。

1. 可做while死循環的條件;

2. 在if分支中作為占位符(即某一分支什麼都不做的時候);

3. 放在必須要有兩元操作的地方作為分隔符,如:: ${username=`whoami`}

4. 在參數替換中為字符串變量賦值,在重定向操作(>)中,把一個文件長度截斷為0(:>>這樣用的時候,目標存在則什麼都不做),這個只能在普通文件中使用,不能在管道,符號鏈接和其他特殊文件中使用;

5. 甚至你可以用來注釋(#後的內容不會被檢查,但:後的內容會被檢查,如果有語句如果出現語法錯誤,則會報錯);

6. 你也可以作為域分隔符,比如環境變量$PATH中,或者passwd中,都有冒號的作為域分隔符的存在;

7. 你也可以將冒號作為函數名,不過這個會將冒號的本來意義轉變(如果你不小心作為函數名,你可以使用unset -f : 來取消function的定義)。

!

感歎號(reverse (or negate) [bang],[exclamation mark])。

取反一個測試結果或退出狀態。

1. 表示反邏輯,比如後面的!=,這個是表示不等於;

2. 表示取反,如:ls a[!0-9] #表示a後面不是緊接一個數字的文件;

3. 在不同的環境裡面,感歎號也可以出現在間接變量引用裡面;

4. 在命令行中,可以用於歷史命令機制的調用,你可以試試!$,!#,或者!-3看看,不過要注意,這點特性不能在腳本文件裡面使用(被禁用)。

*

星號(wildcard/arithmetic operator[asterisk])。

1. 作為匹配文件名擴展的一個通配符,能自動匹配給定目錄下的每一個文件;

2. 正則表達式中可以作為字符限定符,表示其前面的匹配規則匹配任意次;

3. 算術運算中表示乘法。

**

雙星號(double asterisk)。算術運算中表示求冪運算。

?

問號(test operator/wildcard[Question mark])。

1. 表示條件測試;

2. 在雙括號內表示C風格的三元操作符((condition?true-result:false-result));

3. 參數替換表達式中用來測試一個變量是否設置了值;

4. 作為通配符,用於匹配文件名擴展特性中,用於匹配單個字符;

5. 正則表達式中,表示匹配其前面規則0次或者1次。

$

美元符號(Variable substitution[Dollar sign])。

1. 作為變量的前導符,用作變量替換,即引用一個變量的內容,比如:echo $PATH

2. 在正則表達式中被定義為行末(End of line)。

${}

參數替換(Variable substitution)。

用於在字符串中表示變量。

$‘...’

引用內容展開,執行單引號內的轉義內容(單引號原本是原樣引用的),這種方式會將引號內的一個或者多個[\]轉義後的八進制,十六進制值展開到ASCII或Unicode字符。

$*

$@

位置參數(Positional Parameters)。

這個在使用腳本文件的時候,在傳遞參數的時候會用到。兩者都能返回調用腳本文件的所有參數,但$*是將所有參數作為一個整體返回(字符串),而$@是將每個參數作為單元返回一個參數列表。注意,在使用的時候需要用雙引號將$*,$@括住。這兩個變量受到$IFS的影響,如果在實際應用中,要考慮其中的一些細節。

$#

表示傳遞給腳本的參數數量。

$?

此變量值在使用的時候,返回的是最後一個命令、函數、或腳本的退出狀態碼值,如果沒有錯誤則是0,如果為非0,則表示在此之前的最後一次執行有錯誤。

$$

進程ID變量,這個變量保存了運行當前腳本的進程ID值。

()

圓括號(parentheses)。

1, 命令組(Command group)。由一組圓括號括起來的命令是命令組,命令組中的命令實在子shell(subshell)中執行。因為是在子shell內運行,因此在括號外面是沒有辦法獲取括號內變量的值,但反過來,命令組內是可以獲取到外面的值,這點有點像局部變量和全局變量的關系,在實作中,如果碰到要cd到子目錄操作,並在操作完成後要返回到當前目錄的時候,可以考慮使用subshell來處理;

2. 用於數組的初始化。

{x,y,z,...}

花括號擴展(Brace Expansion)。

在命令中可以用這種擴展來擴展參數列表,命令將會依照列表中的括號分隔開的模式進行匹配擴展。注意的一點是,這花括號擴展中不能有空格存在,如果確實有必要空格,則必須被轉義或者使用引號來引用。例子:echo {a,b,c}-{\ d," e",' f'}

{a..z}

在Bash version 3時添加了這種花括號擴展的擴展,可以使用{A..Z}表示A-Z的所有字符列表,這種方式的擴展Mitchell測試了一下,好像僅適用於A-Z,a-z,還有數字{最小..最大}的這種方式擴展。

{}

代碼塊(curly brackets)。

這個是匿名函數,但是又與函數不同,在代碼塊裡面的變量在代碼塊後面仍能訪問。注意:花括號內側需要有空格與語句分隔。另外,在xargs -i中的話,還可以作為文本的占位符,用以標記輸出文本的位置。

{} \;

這個{}是表示路徑名,這個並不是shell內建的,現在接觸到的情況看,好像只用在find命令裡。注意後面的分號,這個是結束find命令中-exec選項的命令序列,在實際使用的時候,要轉義一下以免被shell理解錯誤。

[]

中括號(brackets)。

1. 測試的表示,Shell會測試在[]內的表達式,需要注意的是,[]是Shell內建的測試的一部分,而非使用外部命令/usr/bin/test的鏈接;

2. 在數組的上下文中,表示數組元素,方括號內填上數組元素的位置就能獲得對應位置的內容,如:

  1. Array[1]=xxx
  2. echo ${Array[1]};

3. 表示字符集的范圍,在正表達式中,方括號表示該位置可以匹配的字符集范圍。

[[]]

雙中括號(double brackets)。

這個結構也是測試,測試[[]]之中的表達式(Shell的關鍵字)。這個比單中括號更能防止腳本裡面的邏輯錯誤,比如:&&,||,<,>操作符能在一個[[]]裡面測試通過,但是在[]卻不能通過。[[]]裡面沒有文件名擴展(filename expansion)或是詞分隔符(Word splitting),但是可以用參數擴展(Parameter expansion)和命令替換(command substitution)。不用文件名通配符和像空白這樣的分隔符。注意,這裡面如果出現了八進制,十六進制等,shell會自動執行轉換比較。

$[...]

詞表達表示整數擴展(integer expansion)。

在方括號裡面執行整數表達式。例:

  1. a=3
  2. b=7
  3. echo $[$a+$b]
  4. echo $[$a*$b]
  5. ##返回是10和21
(())

雙括號(double parentheses)。

表示整數擴展(integer expansion)。功能和上面的$[]差不多,但是需要注意的是,$[]是會返回裡面表達式的值的,而(())只是執行,並不會返回值。兩者執行後如果變量值發生變化,都會影響到後繼代碼的運行。可對變量賦值,可以對變量進行一目操作符操作,也可以是二目,三目操作符。

>

&<

>&

>>

<

<>

重定向(redirection)。

scriptname >filename 重定向scriptname的輸出到文件filename中去,如果文件存在則覆蓋;

command &>filename 重定向command的標准輸出(stdout)和標准錯誤(stderr)到文件filename中;

command >&2 把command的標准輸出(stdout)重定向到標准錯誤(stderr)中;

scriptname >>filename 把scriptname的輸出(同>)追加到文件filenmae中,如果文件不存在則創建。

[i]<>filename 打開filename這個文件用來讀或者寫,並且給文件指定i為它的文件描述符(file descriptor),文件不存在就會創建。

(command)>

<(command)

這是進程替換(Process Substitution)。

使用的時候注意,括號和<,>之間是不能有空格的,否則報錯。其作用有點類似通道,但和管道在用法上又有些不同,管道是作為子進程的方式來運行的,這個命令會在/dev/fd/下面產生類似/dev/fd/63,/dev/fd/62這類臨時文件,用來傳遞數據。

Mitchell個人猜測之所以用這種方法來傳遞,是因為前後兩個不屬於同一個進程,因此需要用共享文件的方式來傳遞資料(這麼說其實管道也應該有同樣的文件?)。網上有人說這個只是共享文件而已,但是經過測試,發現雖然有/dev/fd/63這樣的文件產生,但是這個文件其實是指向pipe:[43434]這樣的通道的鏈接。

<<

雙小於號(here-document[double less then marks])。

這個也被稱為Here-document,用來將後繼的內容重定向到左側命令的stdin中。<<可以節省格式化時間,別且使命令執行的處理更容易。在實作的時候只需要輸入<<和終止標志符,而後(一般是回車後)你就可以輸入任何內容,只要在最後的新行中輸入終止標志符,即可完成數據的導入。使用here-document的時候,你可以保留空格,換行等。如果要讓shell腳本更整潔一點,可以在<<和終止符之間放上一個連字符(-)。

<<<

三個小於號(here-strings)。Here-字串和Here-document類似,here-strings語法:command [args] <<<["]$word["];$word會展開並作為command的stdin。

<

>

小於,大於號(ASCII Comparison)。

ASCII比較,進行的是變量的ASCII比較,字串?數字?呃...這個...不就是ASCII比較麼?

\<...\>

詞界符(word boundary)。

這個是用在正則表達式中的一個特殊分隔符,用來標記單詞的分界。比如:the會匹配there,another,them等等,如果僅僅要匹配the,就可以使用這個詞界符,\<the\>就只能匹配the了。

|

管道(pipe)。管道是Linux,Unix都有的概念,是非常基礎,也是非常重要的一個概念。它的作用是將管道前(左邊)的命令產生的輸出(stdout)作為管道後(右邊)的命令的輸入(stdin)。如:ls | wc l,使用管道就可以將命令連接在一起。注意:管道是每一個進程的標准輸出都會作為下一個命令的標准輸入,期間的標准輸出不能跨越管道作為後繼命令的標准輸入,如: cat filename | ls -al | sort 。想想這個的輸出? 同時,管道是以子進程來運行的,所以管道並不能引起變量改變。

>|

強制重定向(force redirection)。

這會強制重寫已經存在的文件。

&

與號(Run job in background[ampersand])。

如果命令後面跟上一個&符號,這個命令將會在後台運行。有的時候,腳本中在一條在後台運行的命令可能會引起腳本掛起,等待輸入,出現這種情況可以在原有的腳本後面使用wait命令來修復。

&&

||

邏輯操作符(logical operator)。

在測試結構中,可以用這兩個操作符來進行連接兩個邏輯值。||是當測試條件有一個為真時返回0(真),全假為假;&&是當測試條件兩個都為真時返回真(0),有假為假。

-

減號,連字符(Hyphen/minus/dash)。

1. 作為選項,前綴[option, prefix]使用。用於命令或者過濾器的選項標志;操作符的前綴。如:

  1. ## COMMAND -[選項列表]
  2. ls -al
  3. sort -dfu $file
  4. set-- $variable
  5. if[ $file -ot $file2 ]
  6. then
  7. echo "$file is older than $file2."
  8. fi

2. 用於stdin或者stdout的重定向的源或目的[dash].在tar沒有bunzip2的程序補丁時,我們可以這樣: bunzip2 linux-2.6.13.tar.bz2 | tar xvf - 。將前面解壓的數據作為tar的標准輸入(這裡使用一個-表示)

注意:在實作的時候,如果文件名是以[-]開頭的,那麼在加上這個作為定向操作符的時候,可能會出錯,此時應該為文件加上合適的前綴路徑,以避免這種情況發生,同樣的,在echo變量的時候,如果變量是以[-]開始,那麼可能也會產生意想不到的結果,為了保險起見,可以使用雙引號引用標量:

  1. var="-n"
  2. echo $var
  3. ## 試試看有什麼輸出?

還有,這種表示方法不是Bash內建的,要達到此點的這種效果,需要看你使用的軟件是否支持這種操作;

3. 表示先前的工作目錄(previous working directory),因此,如果你cd到其他目錄下要放回前一個路徑的時候,可以使用cd -來達到目的,其實,這裡的[-]使用的是環境變量的$OLDPWD,注意:這裡的[-]和前一點是不同的;

4. 減號或者負號,用在算術操作中。

=

等號(Equals)。

1. 賦值操作,給變量賦值,麼有空格在等號兩側;

2. 在比較測試中作為比較符出現,這裡要注意,如果在中括號中作為比較出現,需要有空格符在等號左右兩側。

+

加號(Plus)。

1. 算術操作符,表示加法;

2. 在正則表達式中,表示的是其前的這個匹配規則匹配最少一次;

3.在命令或過濾器中作為選項標記,在某些命令或者內置命令中使用+來啟用某些選項,使用-來禁止;

4. 在參數替換(parameter substitution)中,+前綴表示替代值(當變量為空的時候,使用+後面的值)

%

百分號(modulo[percent sign])。

1.在算術運算中,這個是求模操作符,即兩個數進行除法運算後的余數;

2. 在參數替換(parameter substitution)中,可以作為模式匹配。例子:

  1. p=b*9
  2. var="abcd12345abc479"
  3. echo ${var%p}, ${var%%p}
  4. ##從右邊開始查找(想想從左是那個符號?)
  5. ##任何在b和9之間的內容(含)
  6. ##第一個是找到最短的符合匹配項
  7. ##後一個是找最大符合的匹配項(貪婪匹配?)
~

波浪號(Home directory[tilde])。

這個和內部變量$HOME是一樣的。默認表示當前用戶的家目錄(主目錄),這個和~/效果一致,如果波浪號後面跟用戶名,表示是該用戶的家目錄。

~+

當前的工作目錄(current working directory)。

這個和內置變量$PWD一樣。

~-

前一個工���目錄(previous working directory)。

這個和內部變量$OLDPWD一致,之前的[-]也一樣。

=~

Bash 版本3中有介紹,這個是正則表達式匹配。可用在[[]]測試中,比如:

  1. var="this is a test message."
  2. [["$var"=~ tf*message ]]&& echo "Sir. Found that."|| echo "Sorry Sir. No match be found."
  3. ##你可以修改中間的正則表達式匹配項,正則表達式可以但不一定需要使用雙引號括起來。
^

脫字符(caret)。

1. 在正則表達式中,作為一行的行首(beginning-of-line)位置標志符;

2. 在參數替換(Parameter substitution)中,這個用法有兩種,一個脫字符(${var^}),或兩個(${var^^}),分別表示第一個字母大寫,全部大寫的意思(Bash version >=4)。

空白

空白符(Whitespace)。

空白符不僅僅是指空格(spaces),還包括制表符(tabs),空行(blank lines),或者這幾種的組合。可用做函數的分隔符,分隔命令或變量,空行不會影響腳本的行為,因此可以用它來規劃腳本代碼,以增加可讀性,在內置的特殊變量$IFS可以用來針對某些命令進行輸入的參數進行分割,其默認就是空白符。在字符串或變量中如果有空白符,可以使用引號來規避可能的錯誤。

怎樣,你有多少是了解的呢?Mitchell在開始的Shell腳本時候,發現在這裡面有好多都是不認識呢。

說明:因為涉及到翻譯,文中內容不一定完全翻譯准確,如果你發現有錯誤的地方,還請包涵指正。

參考:

  1. 本文主要內容來源:Advanced Bash-Scripting Guide
  2. 對話 UNIX: !$#@*%
  3. wikipedia的Here文檔

參考內容為本篇成文之際給予Mitchell幫助較大的文章,在整個過程中還有很多網站信息給我提供了幫助,在此對他們的作者的無私貢獻表示感謝!

Copyright © Linux教程網 All Rights Reserved