如何能讓自己的shell顯得不那麼業餘?下面6點實踐一定有用。
畫外音:本篇文章源自Google的一篇實踐,抽取了部分1分鐘能讀完的內容,加入了一些分析。
一、以下面的語句開場
set -o nounset
在默認情況下,遇到不存在的變量,會忽略並繼續執行,而這往往不符合預期,加入該選項,可以避免惡果擴大,終止腳本的執行。
畫外音:有些變量名的手誤,會讓人崩潰的調試半天,通過這個方式,這類手誤秒發現。
set -o errexit
在默認情況下,遇到執行出錯,會跳過並繼續執行,而這往往不符合預期,加入該選項,可以避免惡果擴大,終止腳本的執行。
畫外音:有些Linux命令,例如rm的-f參數可以強制忽略錯誤,此時腳本便無法捕捉到errexit,這樣的參數在腳本里是不推薦使用的。
這兩個選項,都符合fail fast設計理念。
二、封裝函數有必要
別光顧着一溜往下寫,封裝可以提高複用。
如上例:
log()
簡單封裝,能夠省去很多
[$(date +%Y/%m/%d\ %H:%M:%S)]
的重複代碼。
畫外音:這個log()有點意思,學到了嗎?
同時,封裝還能提高代碼的可讀性。
如上例:
ExtractBashComments
比
egrep "^#"
的可讀性就高很多。
畫外音:有些擡槓的朋友會說,不認識英文。
三、使用readonly和local修飾變量
readonly
顧名思義,只讀。
local
函數內變量。
別圖省事,提高安全性的同時,能避免很多讓人崩潰的莫名其妙的錯誤。腳本寫得專不專業,往往不是什麼高深的點,從基本功絕逼能體現。
畫外音:據說,一個C++程序員到什麼水平,從ta代碼裏const的使用頻度,能夠看出來。
四、使用$()代替`(反單引號)
爲什麼?看了上面的例子你就懂了:
(1)$()能夠支持內嵌;
(2)$()不用轉義;
(3)有些字體,`(反單引號)和’(單引號)很像,容易把人搞暈;
五、使用[[]]代替[]
用單中括號:
用雙中括號:
看出差別了麼?[[]]更符合人性編碼:
(1)避免轉義問題;
(2)有不少新功能;
新功能包含但不限於:
|| :邏輯or
&& :邏輯and
< :字符串比較(不需要轉義)
== :通配符(globbing)字符串比較
=~ :正則表達式(regular expression, RegEx)字符串比較
需要注意的是,從bash3.2開始,通配符和正則表達式都不能用引號包裹了(所以,上面的例子,加了引號就是字面比較)。
所以如果表達式裏有空格,必須存儲到一個變量裏,再進行通配符與正則的比較。
六、echo不是唯一的調試方法
可以用-n對腳本進行語法檢查。
可以用-v跟蹤腳本里的每個命令的執行。
可以用-x跟蹤腳本里的每個命令的執行,並附加擴充信息。
當然,也可以在腳本里,添加
set -o verbose
set -o xtrace
來永久指定輸出調試信息。
畫外音:多在自己在機器上試一下就明白了。