!是什么意思!在Shell中执行命令之前?


73

问题在标题中。以感叹号开头的shell命令(shell脚本的一部分)的目的是什么?具体示例:

在foo.sh中:

#!/usr/bin/env bash
set -e
! docker stop foo
! docker rm -f foo
# ... other stuff

我知道没有空格,感叹号用于替换历史记录,并且! <expression>根据手册页可以用于评估“ expr为false时为True ”。但是在示例上下文中,这对我来说没有意义。


8
如果返回码(显示为$?)为0(成功),则将其翻转为1;如果返回码为失败(非零),则将其翻转为0(零)。可能这些命令可能会失败,并且您已-e设置或set -o errexit?无论如何,在这种情况下,应该添加注释。
cdarke

1
@cdarkeset -e忽略退出状态用显式否定的命令!
chepner

Answers:


78

TL; DR:这只是绕过set -e使用它的特定行中的标志。


hek2mgl的正确和有用的答案中添加add 。

你有:

set -e
! command

Bash参考手册→管道描述:

管道中的每个命令都在其自己的子Shell中执行。管道的退出状态是管道中最后一条命令的退出状态(...)。如果保留字为“!” 在流水线之前,退出状态是如上所述退出状态的逻辑取反。在返回值之前,shell等待管道中的所有命令终止。

这意味着!在命令之前会否定其退出状态:

$ echo 23
23
$ echo $?
0
                # But
$ ! echo 23
23
$ echo $?
1

要么:

$ echo 23 && echo "true" || echo "fail"
23
true
$ ! echo 23 && echo "true" || echo "fail"
23
fail

退出状态在许多方面很有用。在您的脚本中,与set -e当命令返回非零状态时可使脚本退出。

因此,当您拥有:

set -e
command1
command2

如果command1返回非零状态,则脚本将完成并且不会继续执行command2

但是,还有一个有趣的要点,如4.3.1内置内置集中所述:

-e

如果可能由单个简单命令(请参见简单命令),列表(请参见列表)或复合命令(请参见复合命令)组成的管道(请参见“管道”)返回非零状态,则立即退出。如果失败的命令紧随一段时间或直到关键字,if语句中测试的一部分,&&或||中执行的任何命令的一部分之后的命令列表的一部分,则外壳程序不会退出。列出除最后一个&&或||之后的命令,管道中除最后一个命令外的任何命令,或者如果命令的返回状态用!反转!。如果除子外壳程序以外的复合命令由于在-e被忽略时命令失败而返回非零状态,则外壳程序不会退出。如果设置了ERR,则会在外壳程序退出之前执行陷阱。


当您具备以下条件时,请考虑所有这些因素:

set -e
! command1
command2

您正在做的是绕过。中的set -e标志command1。为什么?

  • 如果command1运行正常,它将返回零状态。!会否定它,但set -e不会触发退出,因为它来自返回状态,与!反向,如上所述。
  • 如果command1失败,它将返回非零状态。!将对其取反,因此该行最终将返回零状态,并且脚本将正常继续。

31
因此,使用set -e,表示脚本中不允许出现错误,但是使用前导!,则表示异常,表示此命令可以成功执行。
ryenus

5
@ryenus不是。!只需取消命令的退出状态即可。它只是退出状态set -e将被忽略的命令列表中的许多例外之一。
chepner

6
@chepner:我不确定你为什么说“不是真的”。ryenus的评论是正确的。
基思·汤普森

1
@ryenus是的,我有。!不管该-e选项是否有效,都将否定命令的退出状态。它不只是将命令标记为该-e选项的例外。
chepner '17

2
!无论set -e是否设置,@ chpner都确实会否定命令退出状态。但是,要点是关于的上下文中的副作用,!set -e即使命令成功,该脚本也不会使脚本失败。没有这种副作用。当成功命令的退出状态被否定时,脚本将失败!
ryenus

24

如果您不希望脚本在两种情况下(命令错误或成功)都失败,则也可以使用以下替代方法:

set -e
docker stop foo || true

布尔值或true会使管道始终将其0作为返回值。


1
man bash如果保留字!在管道之前,该管道的退出状态是退出状态的逻辑否定。这是否意味着如果命令成功执行,由于否定True,将触发错误?
fedorqui'SO停止伤害

是的,我没有使用!
hek2mgl

@fedorqui我期望相同,但是当我使用简单的演示脚本尝试时,命令成功后它不会终止。例如set -e\n! echo "I am okay"\n\echo "done"贯穿。set -e\n! give_me_an_error\n\echo "done"也贯穿(答案中对此进行了描述)。因此,至少在我的shell中(在macOS上为zsh),它对错误宽容并且不会干扰成功的exit 0命令。
sthzg

12
@sthzg好,我找到了。从猛砸参考手册→4.3.1 set内建-e如果(...)命令的返回状态被倒置shell不会退出!
fedorqui'SO停止伤害

1
@fedorqui很好的收获,很高兴知道!我建议将其回答
hek2mgl
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.