调试 Bash 脚本的方法详解

发布时间:2020-03-18编辑:脚本学堂
本文介绍下,在linux中如何调试bash脚本的一些方法,供大家学习参考。

用 Bash 写的脚本,你知道如何调试吗?
其它,它与python,Perl 等解释型语言一样,都是可以调试的。

新建一个名为 servinfo 的脚本并增加可执行权限:
 

复制代码 代码示例:
$ vi servinfo
#!/bin/bash
echo "hostname: $(hostname)"
echo "Date: $(date)"
echo "Kernel: $(uname -mrs)"

加可执行权限:
 

复制代码 代码示例:
$ chmod +x servinfo

用 bash -x 来调试上述脚本,Bash 先打印出每行脚本,再打印出每行脚本的执行结果:
 

复制代码 代码示例:
$ bash -x servinfo
++ hostname
+ echo 'Hostname: vpsee'
Hostname: vpsee
++ date
+ echo 'Date: Thu Sep  3 19:33:48 SAST 2009'
Date: Thu Sep  3 19:33:48 SAST 2009
++ uname -mrs
+ echo 'Kernel: linux 2.6.18-128.4.1.el5 i686'
Kernel: Linux 2.6.18-128.4.1.el5 i686

如果想同时打印行号的话,可以在脚本开头加上:
 

复制代码 代码示例:
export PS4='+${BASH_SOURCE}:${LINENO}:${FUNCNAME[0]}: '

执行结果:
 

复制代码 代码示例:
$ bash -x servinfo
+ export 'PS4=+${BASH_SOURCE}:${LINENO}:${FUNCNAME[0]}: '
+ PS4='+${BASH_SOURCE}:${LINENO}:${FUNCNAME[0]}: '
++4:5:: hostname
+4:5:: echo 'Hostname: vpsee'
Hostname: vpsee
++4:6:: date
+4:6:: echo 'Date: Thu Sep  3 19:42:06 SAST 2009'
Date: Thu Sep  3 19:42:06 SAST 2009
++4:7:: uname -mrs
+4:7:: echo 'Kernel: Linux 2.6.18-128.4.1.el5 i686'
Kernel: Linux 2.6.18-128.4.1.el5 i686

如果只想调试其中几行脚本的话可以用 set -x 和 set +x 把要调试的部分包含进来:
 

复制代码 代码示例:

#!/bin/bash

echo "Hostname: $(hostname)"
set -x
echo "Date: $(date)"
set +x
echo "Kernel: $(uname -mrs)"

这时就可以直接运行脚本,不需要执行 bash -x 了:
 

复制代码 代码示例:
$ ./servinfo
Hostname: vpsee
++ date
+ echo 'Date: Thu Sep  3 19:46:53 SAST 2009'
Date: Thu Sep  3 19:46:53 SAST 2009
+ set +x
Kernel: Linux 2.6.18-128.4.1.el5 i686

日志输出

跟踪日志有时候太多了,多得都受不了,而且,输出的内容很难阅读。一般来说,我们很多时候只关心于条件表达式,变量值,或是函数调用,或是循环等。。在这种情况下,log一些感兴趣的特定的信息,可能会更好。

使用log前,先写一个函数:
 

复制代码 代码示例:
_log() {
    if [ "$_DEBUG" == "true" ]; then
        echo 1>&2 "$@"
    fi
}

于是,就可以在脚本中这样使用:
 

复制代码 代码示例:
_log "Copying files..."
cp src/* dst/

可以看到,上面那个_log函数,需要检查一个_DEBUG 变量,只有这个变量是真,才会真正开发输出日志。这样,你就只需要控制这个开关,而不需要删除你的debug信息。
$ _DEBUG=true ./example_script.sh

如果要调试一个非常复杂的 Bash 脚本的话,建议用专门的调试工具,比如:bashdb