我在编写shell脚本方面是个新手。如果命令失败,我想打印一条消息并退出脚本。我试过了:
my_command && (echo 'my_command failed; exit)
但这并不奏效。它继续执行脚本中这一行之后的指令。我使用Ubuntu和bash。
我在编写shell脚本方面是个新手。如果命令失败,我想打印一条消息并退出脚本。我试过了:
my_command && (echo 'my_command failed; exit)
但这并不奏效。它继续执行脚本中这一行之后的指令。我使用Ubuntu和bash。
当前回答
还要注意,每个命令的退出状态都存储在shell变量$?,可在执行命令后立即查看。非零状态表示失败:
my_command
if [ $? -eq 0 ]
then
echo "it worked"
else
echo "it failed"
fi
其他回答
如果您希望脚本中的所有命令都具有这种行为,只需添加
set -e
set -o pipefail
在脚本的开头。这对选项告诉bash解释器,当命令返回非零退出码时,就退出。(更多关于为什么需要pipefail的详细信息,请参见http://petereisentraut.blogspot.com/2010/11/pipefail.html)
但是,这并不允许您打印退出消息。
内置的trap shell允许捕获信号和其他有用的条件,包括失败的命令执行(即非零返回状态)。因此,如果你不想显式地测试每个命令的返回状态,你可以输入“你的shell代码”ERR,当命令返回非零状态时,shell代码将被执行。例如:
返回脚本失败;exit 1" ERR
注意,与捕获失败命令的其他情况一样,管道需要特殊处理;上面不会捕获false | true。
你也可以使用,如果你想保留退出错误状态,并有一个可读的文件,每行一个命令:
my_command1 || exit
my_command2 || exit
但是,这将不会打印任何额外的错误消息。但在某些情况下,失败的命令仍然会打印错误。
我把下面这个习语拼凑起来了:
echo "Generating from IDL..."
idlj -fclient -td java/src echo.idl
if [ $? -ne 0 ]; then { echo "Failed, aborting." ; exit 1; } fi
echo "Compiling classes..."
javac *java
if [ $? -ne 0 ]; then { echo "Failed, aborting." ; exit 1; } fi
echo "Done."
在每个命令的前面加上一个有信息的回显,并在每个命令后面加上同样的回显 如果[$?-ne 0];…线。(当然,如果您愿意,您可以编辑该错误消息。)
直接使用exit可能有点棘手,因为脚本可能来自其他地方(例如,从终端)。我更喜欢使用带有set -e的subshell(加上错误应该进入cerr,而不是cout):
set -e
ERRCODE=0
my_command || ERRCODE=$?
test $ERRCODE == 0 ||
(>&2 echo "My command failed ($ERRCODE)"; exit $ERRCODE)