TL; DR:这只是绕过
set -e您使用它的特定行中的标志。
在hek2mgl的正确和有用的答案中添加add 。
你有:
set -e! command
Bash参考手册→管道描述:
管道中的每个命令都在其自己的子Shell中执行。管道的退出状态是管道中最后一条命令的退出状态(…)。 如果保留字为“!”
在管道之前,退出状态是如上所述 退出状态的逻辑取反 。在返回值之前,shell等待管道中的所有命令终止。
这意味着
!在命令之前会否定其退出状态:
$ echo 2323$ echo $?0 # But$ ! echo 2323$ echo $?1
要么:
$ echo 23 && echo "true" || echo "fail"23true$ ! echo 23 && echo "true" || echo "fail"23fail
退出状态在许多方面很有用。在脚本中,
set -e当命令返回非零状态时,与一起使用可使脚本退出。
因此,当您拥有:
set -ecommand1command2
如果
command1返回非零状态,则脚本将完成并且不会继续进行
command2。
但是,还有一个有趣的要点,如4.3.1内置内置集中所述:
-e
如果可能由单个简单命令(请参见简单命令),列表(请参见列表)或复合命令(请参见复合命令)组成的管道(请参见“管道”)返回非零状态,则立即退出。如果失败的命令紧随一段时间或直到关键字,if语句中测试的一部分,&&或||中执行的任何命令的一部分之后的命令列表的一部分,则
外壳程序不会退出。 列出最后一个&&或||之后的命令,管道中最后一个命令之外的任何命令,或者 如果命令的返回状态用!反转!
。如果除子外壳程序以外的复合命令由于在-e被忽略时命令失败而返回非零状态,则外壳程序不会退出。退出外壳之前,将执行ERR上的陷阱(如果已设置)。
当您具备以下条件时,请考虑所有这些因素:
set -e! command1command2
您正在做的是绕过。中的
set -e标志
command1。为什么?
- 如果
command1
运行正常,它将返回零状态。!
会否定它,但set -e
不会触发退出,因为如上所述,它来自返回状态,并以!取反。 - 如果
command1
失败,它将返回非零状态。!
会将其取反,因此该行最终将返回零状态,并且脚本将正常继续。



