我有一个Bash shell脚本,它调用许多命令。

如果任何命令返回非零值,我想让shell脚本自动退出并返回值为1。

如果不显式地检查每个命令的结果,这可能吗?

例如,

dosomething1
if [[ $? -ne 0 ]]; then
    exit 1
fi

dosomething2
if [[ $? -ne 0 ]]; then
    exit 1
fi

当前回答

美元吗?很少需要变量。伪成语命令;如果[$?-eq 0];那么X;Fi应该总是写成command;那么X;fi。

$?Is required是指需要检查多个值:

command
case $? in
  (0) X;;
  (1) Y;;
  (2) Z;;
esac

或者什么时候$?需要重用或以其他方式操作:

if command; then
  echo "command successful" >&2
else
  ret=$?
  echo "command failed with exit code $ret" >&2
  exit $ret
fi

其他回答

运行时带-e或者在顶部设置-e。

再看看set -u。

示例中的if语句是不必要的。就像这样做:

dosomething1 || exit 1

如果你采纳了Ville Laurikari的建议并使用set -e,那么对于某些命令你可能需要使用这个:

dosomething || true

|| true将使命令管道即使在命令失败时也具有真实的返回值,因此-e选项不会终止脚本。

对已接受的答案补充:

请记住,set -e有时是不够的,特别是如果你有管道。

例如,假设您有这个脚本

#!/bin/bash
set -e 
./configure  > configure.log
make

... 按预期工作:配置中的错误将中止执行。

明天你要做一个看似微不足道的改变:

#!/bin/bash
set -e 
./configure  | tee configure.log
make

... 但现在它不管用了。这里解释了这一点,并提供了一个解决方案(仅Bash):

#!/bin/bash
set -e 
set -o pipefail

./configure  | tee configure.log
make

如果出现错误,下面的脚本将打印一个红色错误消息并退出。 把这个放在你的bash脚本的顶部:

# BASH error handling:
#   exit on command failure
set -e
#   keep track of the last executed command
trap 'LAST_COMMAND=$CURRENT_COMMAND; CURRENT_COMMAND=$BASH_COMMAND' DEBUG
#   on error: print the failed command
trap 'ERROR_CODE=$?; FAILED_COMMAND=$LAST_COMMAND; tput setaf 1; echo "ERROR: command \"$FAILED_COMMAND\" failed with exit code $ERROR_CODE"; put sgr0;' ERR INT TERM

像这样的表达式

dosomething1 && dosomething2 && dosomething3

当其中一个命令返回非零值时将停止处理。例如,下面的命令永远不会打印"done":

cat nosuchfile && echo "done"
echo $?
1