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

时间:2017-11-13 09:47:28

标签: linux bash shell unix sh

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

在foo.sh中:

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

我知道如果没有空格,感叹号将用于历史记录替换,! <expression>根据man page可用于评估&#34; 如果expr为false则为真&#34 ;.但是在对我来说没有意义的示例上下文中。

3 个答案:

答案 0 :(得分:77)

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

添加hek2mgl's correct and useful answer

你有:

set -e
! command

Bash Reference Manual → Pipelines描述:

  

管道中的每个命令都在其自己的子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 The Set Builtin中描述:

  

<强> -e

     

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

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

set -e
! command1
command2

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

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

答案 1 :(得分:23)

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

set -e
docker stop foo || true

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

答案 2 :(得分:1)

“!”意思是“不”,所以当你把它放在一个命令之前然后你运行“$?”它将回显1.这意味着它失败而不是0,这意味着成功。