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,如:
#!/bin/bash
. data-file
#包含data-file;
2. 作為文件名的一部分,在文件名的開頭,表示該文件為隱藏文件,ls一般不顯示出來(ls -a 可以顯示);
3. 作為目錄名,一個點代表當前目錄,兩個點號代表上層目錄(當前目錄的父目錄)。注意,兩個以上的點不出現,除非你用引號(單/雙)包圍作為點號字符本身;
4. 正則表達式中,點號表示任意一個字符。
"
雙引號(partial quoting [double quote])。
部分引用。雙引號包圍的內容可以允許變量擴展,也允許轉義字符的存在。如果字符串內出現雙引號本身,需要轉義,因此不一定雙引號是成對的。
'
單引號(full quoting [single quote])。
單引號括住的內容,被視為單一字符串,引號內的禁止變量擴展,所有字符均作為字符本身處理(除單引號本身之外),單引號必須成對出現。
,
逗號(comma operator [comma])。
1. 用在連接一連串的數學表達式中,這串數學表達式均被求值,但只有最後一個求值結果被返回。如:
#!/bin/bash
let t1=((a=5+1, b=7+2))
echo t1=$t1, a=$a, b=$b
## 這個$t1=$b;
2. 用於參數替代中,表示首字母小寫,如果是兩個逗號,則表示全部小寫,注意,這個特性在bash version 4的時候被添加的。例子:
a="ATest"
echo ${a,}
echo ${a,,}
## 前面輸出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. 在數組的上下文中,表示數組元素,方括號內填上數組元素的位置就能獲得對應位置的內容,如:
Array[1]=xxx
echo ${Array[1]};
3. 表示字符集的范圍,在正表達式中,方括號表示該位置可以匹配的字符集范圍。
[[]]
雙中括號(double brackets)。
這個結構也是測試,測試[[]]之中的表達式(Shell的關鍵字)。這個比單中括號更能防止腳本裡面的邏輯錯誤,比如:&&,||,<,>操作符能在一個[[]]裡面測試通過,但是在[]卻不能通過。[[]]裡面沒有文件名擴展(filename expansion)或是詞分隔符(Word splitting),但是可以用參數擴展(Parameter expansion)和命令替換(command substitution)。不用文件名通配符和像空白這樣的分隔符。注意,這裡面如果出現了八進制,十六進制等,shell會自動執行轉換比較。
$[...]
詞表達表示整數擴展(integer expansion)。
在方括號裡面執行整數表達式。例:
a=3
b=7
echo $[$a+$b]
echo $[$a*$b]
##返回是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]使用。用於命令或者過濾器的選項標志;操作符的前綴。如:
## COMMAND -[選項列表]
ls -al
sort -dfu $file
set-- $variable
if[ $file -ot $file2 ]
then
echo "$file is older than $file2."
fi
2. 用於stdin或者stdout的重定向的源或目的[dash].在tar沒有bunzip2的程序補丁時,我們可以這樣: bunzip2 linux-2.6.13.tar.bz2 | tar xvf -
。將前面解壓的數據作為tar的標准輸入(這裡使用一個-表示)
注意:在實作的時候,如果文件名是以[-]開頭的,那麼在加上這個作為定向操作符的時候,可能會出錯,此時應該為文件加上合適的前綴路徑,以避免這種情況發生,同樣的,在echo變量的時候,如果變量是以[-]開始,那麼可能也會產生意想不到的結果,為了保險起見,可以使用雙引號引用標量:
var="-n"
echo $var
## 試試看有什麼輸出?
還有,這種表示方法不是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)中,可以作為模式匹配。例子:
p=b*9
var="abcd12345abc479"
echo ${var%p}, ${var%%p}
##從右邊開始查找(想想從左是那個符號?)
##任何在b和9之間的內容(含)
##第一個是找到最短的符合匹配項
##後一個是找最大符合的匹配項(貪婪匹配?)
~
波浪號(Home directory[tilde])。
這個和內部變量$HOME是一樣的。默認表示當前用戶的家目錄(主目錄),這個和~/效果一致,如果波浪號後面跟用戶名,表示是該用戶的家目錄。
~+
當前的工作目錄(current working directory)。
這個和內置變量$PWD一樣。
~-
前一個工���目錄(previous working directory)。
這個和內部變量$OLDPWD一致,之前的[-]也一樣。
=~
Bash 版本3中有介紹,這個是正則表達式匹配。可用在[[]]測試中,比如:
var="this is a test message."
[["$var"=~ tf*message ]]&& echo "Sir. Found that."|| echo "Sorry Sir. No match be found."
##你可以修改中間的正則表達式匹配項,正則表達式可以但不一定需要使用雙引號括起來。
^
脫字符(caret)。
1. 在正則表達式中,作為一行的行首(beginning-of-line)位置標志符;
2. 在參數替換(Parameter substitution)中,這個用法有兩種,一個脫字符(${var^}
),或兩個(${var^^}
),分別表示第一個字母大寫,全部大寫的意思(Bash version >=4)。
空白
空白符(Whitespace)。
空白符不僅僅是指空格(spaces),還包括制表符(tabs),空行(blank lines),或者這幾種的組合。可用做函數的分隔符,分隔命令或變量,空行不會影響腳本的行為,因此可以用它來規劃腳本代碼,以增加可讀性,在內置的特殊變量$IFS可以用來針對某些命令進行輸入的參數進行分割,其默認就是空白符。在字符串或變量中如果有空白符,可以使用引號來規避可能的錯誤。
怎樣,你有多少是了解的呢?Mitchell在開始的Shell腳本時候,發現在這裡面有好多都是不認識呢。
說明:因為涉及到翻譯,文中內容不一定完全翻譯准確,如果你發現有錯誤的地方,還請包涵指正。
參考:
參考內容為本篇成文之際給予Mitchell幫助較大的文章,在整個過程中還有很多網站信息給我提供了幫助,在此對他們的作者的無私貢獻表示感謝!