文章转载自微信公众号 架构师之路,侵删。
一、以下面的语句开场
- 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
来永久指定输出调试信息。
画外音:多在自己在机器上试一下就明白了
补充一下:对于bash的调试来讲,使用vscode的bash插件功能可以很方便的进行语法的检查和单点调试的功能。