許多人用Shell腳本完成一些簡(jiǎn)單任務(wù),而且變成了他們生命的一部分。不幸的是,shell腳本在運(yùn)行異常時(shí)會(huì)受到非常大的影響。在寫(xiě)腳本時(shí)將這類(lèi)問(wèn)題最小化是十分必要的。本文中我將介紹一些讓bash腳本變得健壯的技術(shù)。
使用set -u
你因?yàn)闆](méi)有對(duì)變量初始化而使腳本崩潰過(guò)多少次?對(duì)于我來(lái)說(shuō),很多次。
代碼如下:
chroot=$1
...
rm -rf $chroot/usr/share/doc
如果上面的代碼你沒(méi)有給參數(shù)就運(yùn)行,你不會(huì)僅僅刪除掉chroot中的文檔,而是將系統(tǒng)的所有文檔都刪除。那你應(yīng)該做些什么呢?好在bash提供了set -u,當(dāng)你使用未初始化的變量時(shí),讓bash自動(dòng)退出。你也可以使用可讀性更強(qiáng)一點(diǎn)的set -o nounset。
代碼如下:
david% bash /tmp/shrink-chroot.sh
$chroot=
david% bash -u /tmp/shrink-chroot.sh
/tmp/shrink-chroot.sh: line 3: $1: unbound variable
david%
使用set -e
你寫(xiě)的每一個(gè)腳本的開(kāi)始都應(yīng)該包含set -e。這告訴bash一但有任何一個(gè)語(yǔ)句返回非真的值,則退出bash。使用-e的好處是避免錯(cuò)誤滾雪球般的變成嚴(yán)重錯(cuò)誤,能盡早的捕獲錯(cuò)誤。更加可讀的版本:set -o errexit
使用-e把你從檢查錯(cuò)誤中解放出來(lái)。如果你忘記了檢查,bash會(huì)替你做這件事。不過(guò)你也沒(méi)有辦法使用$?來(lái)獲取命令執(zhí)行狀態(tài)了,因?yàn)閎ash無(wú)法獲得任何非0的返回值。你可以使用另一種結(jié)構(gòu):
代碼如下:
command
if [ "$?"-ne 0]; then echo "command failed"; exit 1; fi
可以替換成:
代碼如下:
command || { echo "command failed"; exit 1; }
或者使用:
代碼如下:
if ! command; then echo "command failed"; exit 1; fi
如果你必須使用返回非0值的命令,或者你對(duì)返回值并不感興趣呢?你可以使用 command || true ,或者你有一段很長(zhǎng)的代碼,你可以暫時(shí)關(guān)閉錯(cuò)誤檢查功能,不過(guò)我建議你謹(jǐn)慎使用。
代碼如下:
set +e
command1
command2
set -e
相關(guān)文檔指出,bash默認(rèn)返回管道中最后一個(gè)命令的值,也許是你不想要的那個(gè)。比如執(zhí)行 false | true 將會(huì)被認(rèn)為命令成功執(zhí)行。如果你想讓這樣的命令被認(rèn)為是執(zhí)行失敗,可以使用 set -o pipefail
程序防御 - 考慮意料之外的事
你的腳本也許會(huì)被放到“意外”的賬戶下運(yùn)行,像缺少文件或者目錄沒(méi)有被創(chuàng)建等情況。你可以做一些預(yù)防這些錯(cuò)誤事情。比如,當(dāng)你創(chuàng)建一個(gè)目錄后,如果父目錄不存在,mkdir 命令會(huì)返回一個(gè)錯(cuò)誤。如果你創(chuàng)建目錄時(shí)給mkdir命令加上-p選項(xiàng),它會(huì)在創(chuàng)建需要的目錄前,把需要的父目錄創(chuàng)建出來(lái)。另一個(gè)例子是 rm 命令。如果你要?jiǎng)h除一個(gè)不存在的文件,它會(huì)“吐槽”并且你的腳本會(huì)停止工作。(因?yàn)槟闶褂昧?e選項(xiàng),對(duì)吧?)你可以使用-f選項(xiàng)來(lái)解決這個(gè)問(wèn)題,在文件不存在的時(shí)候讓腳本繼續(xù)工作。
新聞熱點(diǎn)
疑難解答
網(wǎng)友關(guān)注