如何在shell中处理异常?
我们可以编辑代码处理这种异常,编写代码如下;
#!/bin/bash
TEMP_FIL;
=/tmp/printfile.txt
trap "rm $TEMP_FILE;
exit" SIGHUP SIGINT SIGTERM
pr $1 > $TEMP_FILE
echo -n "Print file? [y/n]:
"
read
if [ "$REPLY" = "y" ];
then
lpr $TEMP_FILE
fi
rm $TEMP_FILE
read
if [ "$REPLY" = "y" ];
then
lpr $TEMP_FILE
fi
rm $TEMP_FILE
返回值
要判断一段代码是否出现了异常,一个最基本的判断就是对他返回值的判断。在shell中,我们往往规定0为正常,一切非0返回值则为不正常。但往往我们在写shell脚本的时候,忽略对于返回值的判断。我们看一个很基本的shell程序
#!/bin/sh
cd /home/xxxx/
rm -rf *
这个脚本的意思很简单,就是cd到某一个目录下,然后将该目录下所有的内容都删除。首先,rm这种东西出现在脚本中,就是一个很危险的操作,而这个程序的关键之处还在于,并没有对第一行shell的返回值进行任何的判断,也就是说对于cd那行代码无论执行失败与否,都会去执行下面的那段rm,试想如果在某些情况下cd那段代码失败了之后,会出现多么可怕的后果。所以,我们应该对于cd的代码做返回值的判断。
#!/bin/sh
cd /home/xxxx/
if [ "$?"= "0" ]; then
rm -rf *
else
echo "cannot change directory" 1>&2
exit 1
fi
$?这个常量代表的就是上一段shell的返回值。这个我在前面一片文章里也提到过shell中的trap和expr。这样写的话,就要比先前的程序安全多了,如果没有cd到相应目录,则不会去执行删除操作。
当然,程序里这样写是有些负责了,其实你也可以这样写:
if cd /home/xxxx/ ; then
rm -rf *
else
echo "cannot change directory" 1>&2
exit 1
这是if的另一种用法。这样写,就要比刚才的好多了。不过其实这样写,也比较麻烦,其实你还可以这样来写:
cd /home/xxxx/ && rm -rf *
这个&&符大家肯定不陌生,那这样来写,是否就可以保证了程序的安全性了呢?下面就来讲一下&&和||
&&和||
对于一个shell程序, shell1 && shell2 ,如果是用&&符连接的,那只有在shell1返回0(即正常)时,shell2才会执行,否则shell2根本就不执行,所以前面说得最后一种cd&&rm的这种做法是可行的,而且是安全的。那||呢,对于shell1||shell2,只有在shell1执行失败时,shell2才会执行,否则shell2是不执行得。所以,我们可以这样来写:
cd /home/xxxx || error_exit "Cannot change directory"
rm -rf *
这样,在cd /home/xxxx失败时,就会进行error_exit这个函数的分支。那error_exit这个是啥函数呢。。哦,其实,这只是一个自定义的失败处理函数而已。一个比较简单的定义,当然,这也是处理程序异常的一个方式。因为总不能每次异常,都去手动写个echo failed exit等等,所以有个统一的函数处理会比较方便。
function error_exit {
echo "$1" 1>&2
exit 1
}
cd /home/xxxx || error_exit "failed"
rm -rf *
如上,这是一个非常简单的异常处理函数,在异常时,可以去主动调用error_exit,当然,你调用的时候,可以补充上行号信息。比如:
cd /home/xxxx || error_exit "$LINENO: failed"