bash 手册指出:
eval [arg ...]
The args are read and concatenated together into a single com-
mand. This command is then read and executed by the shell, and
its exit status is returned as the value of eval. If there are
no args, or only null arguments, eval returns 0.
我尝试
eval `nonsense`
echo $?
结果是0
。
而当我单独执行反引号命令时:
`nonsense`
echo $?
结果是127
。
根据 bash 手册中所写的内容,我希望在将反引号作为参数时eval
返回。127
nonsense
如何获取参数的退出状态eval
?
答案1
当您执行以下操作时 -
`nonsense`
echo $?
您基本上是在问“当我尝试获取无意义的命令的输出时告诉我退出状态”,答案是“找不到命令”或 127
但是当你执行以下操作时
eval `nonsense`
echo $?
您问“当我评估空字符串时告诉我 eval 的退出状态”(无意义命令的输出),这等于eval
不带参数运行。
eval
无参数运行没有问题,退出状态变为0
答案2
事实上,更多的是:
$ `nonsense`
bash: nonsense: command not found
$ echo "$?"
127
这令人惊讶。
我们要求bash
在 的 stdout 上运行 split+glob 运算符的结果命令nonsense
。由于nonsense
不产生输出,因此它不运行命令,因此您可能认为退出状态应该为 0。
但实际上,当一个简单的命令行没有参数,只有赋值或重定向时,退出状态是运行的赋值和正常单词(不在重定向目标中)中的最后一个命令替换的退出状态(尽管重定向失败也会影响退出状态)。
这对于作业特别有用。
在:
output=$(grep pattern file)
status=$?
您可以获取 的输出和退出状态,如果是该非命令的退出状态,则grep
无法获取该输出和退出状态。$?
在:
output=$(cmd1) cmd2
即同时存在赋值词和参数词的地方, 的退出状态cmd1
被忽略。$?
将包含 的退出状态cmd2
。
并且,也$output
将仅设置为cmd2
仅。例外情况是当cmd2
是一个特殊的内置函数时。
eval
是这样一个特别内置。
$ a=0; a=1 eval; echo "$a"
1
在bash
大多数现代 POSIX shell 中。
a=`exit 5` eval; echo "$?"
或者
eval `exit 5`; echo "$?"
eval
将输出 0,因为它是不带参数运行的结果。但在 Bourne shell 或 ksh88 中情况并非如此,对于特殊的内置命令,您将获得exit 5
那里的退出状态。
在这些 shell 中,您还会发现:
$ a=`exit 3` set x; echo "$?"
3
set
另一个特殊的内置函数也是如此。
.
是另一个特殊的内置函数。在 Bourne shell 和 ksh88 中:
$ . /some/file `exit 4`; echo "$?"
4
(只要/some/file
不运行任何命令)