什么意思啊!在shell中的命令之前?

sth*_*hzg 72 unix linux bash shell sh

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

在foo.sh中:

#!/usr/bin/env bash
set -e
! docker stop foo
! docker rm -f foo
# ... other stuff
Run Code Online (Sandbox Code Playgroud)

我知道如果没有空格,感叹号将用于历史记录替换,并且! <expression>根据手册页可用于评估" 如果expr为false则为True ".但是在对我来说没有意义的示例上下文中.

fed*_*qui 78

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


添加添加到hek2mgl的正确和有用的答案.

你有:

set -e
! command
Run Code Online (Sandbox Code Playgroud)

Bash参考手册→管道描述:

管道中的每个命令都在其自己的子shell中执行.管道的退出状态是管道中最后一个命令的退出状态(...).如果保留字'!' 在管道之前,退出状态是如上所述的退出状态的逻辑否定.shell在返回值之前等待管道中的所有命令终止.

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

$ echo 23
23
$ echo $?
0
                # But
$ ! echo 23
23
$ echo $?
1
Run Code Online (Sandbox Code Playgroud)

要么:

$ echo 23 && echo "true" || echo "fail"
23
true
$ ! echo 23 && echo "true" || echo "fail"
23
fail
Run Code Online (Sandbox Code Playgroud)

退出状态在很多方面都很有用.在您的脚本中,与set -e命令一起返回非零状态时,脚本将一起使用.

因此,当你有:

set -e
command1
command2
Run Code Online (Sandbox Code Playgroud)

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

但是,还有一个值得一提的地方,如4.3.1 The Set Builtin中所述:

-e

如果管道(请参阅管道)(可能包含单个简单命令(请参阅简单命令)),列表(请参阅列表)或复合命令(请参阅复合命令)返回非零状态,请立即退出.如果失败的命令是紧跟在while或until关键字之后的命令列表的一部分,if语句中的部分测试,在&&或||中执行的任何命令的一部分,则shell不会退出 列表除了最后的&&或||之后的命令,管道中的任何命令但是最后一个,或者命令的返回状态是否被反转!.如果子shell以外的复合命令返回非零状态,因为在忽略-e时命令失败,则shell不会退出.ERR上的陷阱(如果已设置)将在shell退出之前执行.


考虑到所有这些因素,当您有:

set -e
! command1
command2
Run Code Online (Sandbox Code Playgroud)

你正在做的是绕过set -e国旗command1.为什么?

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

  • 所以使用`set -e`,它意味着**在脚本中没有允许错误**,但是使用前导**`!`**,它标记了一个异常,这意味着该命令可以正常失败. (29认同)
  • @chepner:我不确定你为什么说"不是真的.".ryenus的评论是正确的. (6认同)
  • @ryenus不是真的.`!`只是否定命令的退出状态.它只是退出状态`set -e`将忽略的命令列表中的众多异常之一. (5认同)
  • @chpner,`!`单独确实否定了命令退出状态,无论是否设置了set -e`.但是这里的要点是````在`set -e`**的上下文中的**副作用,即使命令成功也不会使脚本失败.没有这种副作用.当成功命令的退出状态被`!`否定时,脚本将失败. (2认同)

hek*_*mgl 23

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

set -e
docker stop foo || true
Run Code Online (Sandbox Code Playgroud)

boolean 或true使管道始终具有0返回值.

  • @sthzg好的我找到了.从[Bash参考手册→4.3.1 Set Builtin](https://www.gnu.org/software/bash/manual/bashref.html#The-Set-Builtin)上的`-e`:_ shell不退出如果(...)命令的返回状态正在反转!_. (12认同)