繁体   English   中英

!的含义是什么! 在Shell中执行命令之前?

[英]What's the meaning of a ! before a command in the shell?

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

在foo.sh中:

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

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

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

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

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

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

set -e
docker stop foo || true

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

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM