如何在shell中处理异常(Part 1)

本文来自于我的博客 http://www.log4myself.com/archives/339

 

本来就想今天写写在shell中如何处理异常,结果看了一天《复活密码》,然后晚上又在看《危机边缘》,结果就到了现在了。。哎,罪过罪过啊,算了,废话不多说了,开始写。

前言

似乎好像大概有句话是这么说得,好程序与坏程序之间的区别就在于它的鲁棒性,也就是在异常情况下该程序是否还是在可hold住状态,能否不死,不崩溃,或者不做出一些超出预期的事情。那要做好这些,自然而然就要学会如何去处理异常。平时写php或者java程序等等的时候,很多人都会去注重对于异常的处理,比如try..catch等等,但往往在写一些脚本的时候,忽视了对于异常的判断。本文主要就是写如何在shell中去处理异常。因为今天太晚了,所以我就先写第一部分,后面再写第二部分。

返回值

要判断一段代码是否出现了异常,一个最基本的判断就是对他返回值的判断。在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"

上面讲得只是一些简单的异常处理的方式,其实还有其他方式,比如我在上篇文章shell中的trap和expr讲到的trap方式,其实也正是因为我上面文章写了trap,所以才想写这篇文章来把处理shell的异常给整理出来,当然也是为大家提个醒,在shell中,也是要处理异常,需要有这样一个意识。今天太晚了,先睡觉了,明早还要早起T.T ,下篇文章,PART 2中,我会详细总结下trap的捕获异常的方式。

安。

你可能感兴趣的:(shell)