如何测试命令是否输出空字符串?


当前回答

到目前为止给出的所有答案都处理终止并输出非空字符串的命令。

大多数是在以下意义上被打破的:

它们不能正确地处理只输出换行符的命令; 从Bash≥4.4开始,如果命令输出null字节(因为他们使用命令替换),大多数将发送垃圾邮件标准错误; 大多数人将发出完整的输出流,所以会等到命令结束才回答。有些命令永远不会终止(例如,尝试,yes)。

所以为了解决所有这些问题,并有效地回答下面的问题,

如何测试命令是否输出空字符串?

你可以使用:

if read -n1 -d '' < <(command_here); then
    echo "Command outputs something"
else
    echo "Command doesn't output anything"
fi

您还可以使用read的-t选项添加一些超时,以测试命令是否在给定时间内输出非空字符串。例如,对于2.5秒的超时:

if read -t2.5 -n1 -d '' < <(command_here); then
    echo "Command outputs something"
else
    echo "Command doesn't output anything"
fi

的话。如果您认为需要确定命令是否输出非空字符串,那么您很可能遇到了XY问题。

其他回答

到目前为止给出的所有答案都处理终止并输出非空字符串的命令。

大多数是在以下意义上被打破的:

它们不能正确地处理只输出换行符的命令; 从Bash≥4.4开始,如果命令输出null字节(因为他们使用命令替换),大多数将发送垃圾邮件标准错误; 大多数人将发出完整的输出流,所以会等到命令结束才回答。有些命令永远不会终止(例如,尝试,yes)。

所以为了解决所有这些问题,并有效地回答下面的问题,

如何测试命令是否输出空字符串?

你可以使用:

if read -n1 -d '' < <(command_here); then
    echo "Command outputs something"
else
    echo "Command doesn't output anything"
fi

您还可以使用read的-t选项添加一些超时,以测试命令是否在给定时间内输出非空字符串。例如,对于2.5秒的超时:

if read -t2.5 -n1 -d '' < <(command_here); then
    echo "Command outputs something"
else
    echo "Command doesn't output anything"
fi

的话。如果您认为需要确定命令是否输出非空字符串,那么您很可能遇到了XY问题。

有时您希望保存输出(如果它是非空的),以便将其传递给另一个命令。如果是的话,你可以使用

list=`grep -l "MY_DESIRED_STRING" *.log `
if [ $? -eq 0 ]
then
    /bin/rm $list
fi

这样,如果列表为空,rm命令就不会挂起。

我猜你想要ls -al命令的输出,所以在bash中,你会有这样的东西:

LS=`ls -la`

if [ -n "$LS" ]; then
  echo "there are files"
else
  echo "no files found"
fi

下面是另一种方法,将某些命令的std-out和std-err写入临时文件,然后检查该文件是否为空。这种方法的一个好处是它捕获两个输出,并且不使用子外壳或管道。后面这些方面很重要,因为它们会干扰捕获bash退出处理(例如这里)

tmpfile=$(mktemp)
some-command  &> "$tmpfile"
if [[ $? != 0 ]]; then
    echo "Command failed"
elif [[ -s "$tmpfile" ]]; then
    echo "Command generated output"
else
    echo "Command has no output"
fi
rm -f "$tmpfile"

之前,问题询问如何检查目录中是否有文件。下面的代码实现了这一点,但请参阅rsp的答案以获得更好的解决方案。


空的输出

命令不返回值——它们输出值。您可以使用命令替换来捕获此输出;例如$(ls -A)。你可以像这样在Bash中测试一个非空字符串:

if [[ $(ls -A) ]]; then
    echo "there are files"
else
    echo "no files found"
fi

注意,我使用的是-A而不是-A,因为它省略了当前(.)和父目录(..)的符号项。

注意:正如注释中指出的,命令替换不捕获尾随换行符。因此,如果命令只输出换行,则替换将不会捕获任何内容,并且测试将返回false。虽然不太可能,但在上面的例子中这是可能的,因为一个换行符就是一个有效的文件名!更多信息在这个答案中。


退出代码

如果要检查命令是否成功执行,可以检查$?,其中包含最后一个命令的退出代码(0表示成功,非0表示失败)。例如:

files=$(ls -A)
if [[ $? != 0 ]]; then
    echo "Command failed."
elif [[ $files ]]; then
    echo "Files found."
else
    echo "No files found."
fi

更多信息请点击这里。