写出健壮Bash Shell脚本的一些技巧总结
许多人用Shell脚本完成一些简单任务,而且变成了他们生命的一部分。不幸的是,shell脚本在运行异常时会受到非常大的影响。在写脚本时将这类问题最小化是十分必要的。本文中我将介绍一些让bash脚本变得健壮的技术。
使用set-u
你因为没有对变量初始化而使脚本崩溃过多少次?对于我来说,很多次。
chroot=$1 ... rm-rf$chroot/usr/share/doc
如果上面的代码你没有给参数就运行,你不会仅仅删除掉chroot中的文档,而是将系统的所有文档都删除。那你应该做些什么呢?好在bash提供了set-u,当你使用未初始化的变量时,让bash自动退出。你也可以使用可读性更强一点的set-onounset。
david%bash/tmp/shrink-chroot.sh $chroot= david%bash-u/tmp/shrink-chroot.sh /tmp/shrink-chroot.sh:line3:$1:unboundvariable david%
使用set-e
你写的每一个脚本的开始都应该包含set-e。这告诉bash一但有任何一个语句返回非真的值,则退出bash。使用-e的好处是避免错误滚雪球般的变成严重错误,能尽早的捕获错误。更加可读的版本:set-oerrexit
使用-e把你从检查错误中解放出来。如果你忘记了检查,bash会替你做这件事。不过你也没有办法使用$?来获取命令执行状态了,因为bash无法获得任何非0的返回值。你可以使用另一种结构:
command if["$?"-ne0];thenecho"commandfailed";exit1;fi
可以替换成:
command||{echo"commandfailed";exit1;}
或者使用:
if!command;thenecho"commandfailed";exit1;fi
如果你必须使用返回非0值的命令,或者你对返回值并不感兴趣呢?你可以使用command||true,或者你有一段很长的代码,你可以暂时关闭错误检查功能,不过我建议你谨慎使用。
set+e command1 command2 set-e
相关文档指出,bash默认返回管道中最后一个命令的值,也许是你不想要的那个。比如执行false|true将会被认为命令成功执行。如果你想让这样的命令被认为是执行失败,可以使用set-opipefail
程序防御-考虑意料之外的事
你的脚本也许会被放到“意外”的账户下运行,像缺少文件或者目录没有被创建等情况。你可以做一些预防这些错误事情。比如,当你创建一个目录后,如果父目录不存在,mkdir命令会返回一个错误。如果你创建目录时给mkdir命令加上-p选项,它会在创建需要的目录前,把需要的父目录创建出来。另一个例子是rm命令。如果你要删除一个不存在的文件,它会“吐槽”并且你的脚本会停止工作。(因为你使用了-e选项,对吧?)你可以使用-f选项来解决这个问题,在文件不存在的时候让脚本继续工作。
准备好处理文件名中的空格
有些人从在文件名或者命令行参数中使用空格,你需要在编写脚本时时刻记得这件事。你需要时刻记得用引号包围变量。
if[$filename="foo"];
当$filename变量包含空格时就会挂掉。可以这样解决:
if["$filename"="foo"];
使用$@变量时,你也需要使用引号,因为空格隔开的两个参数会被解释成两个独立的部分。
david%foo(){foriin$@;doecho$i;done};foobar"bazquux" bar baz quux
david%foo(){foriin"$@";doecho$i;done};foobar"bazquux" bar bazquux