linux – Bash while while循环意外停止

我正在分析两个脚本,其中包含一些我不理解的行为:

#/bin/bash
tijd=${1-60}
oud=`ls -l $MAIL`
while : ; do
   nieuw=`ls -l $MAIL`
   echo $oud $nieuw
   sleep $tijd
done | { read a b rest ; echo $a ; echo $b ; echo $rest ; }

此脚本中的while循环在一次迭代后停止.

#/bin/bash
tijd=${1-60}
oud=`ls -l $MAIL`
while : ; do
   nieuw=`ls -l $MAIL`
   echo $oud $nieuw
   sleep $tijd
done | cat

此脚本中的while循环是无限的.

有什么不同?我认为它与管道和括号有关,但我无法解释它.

解决方法

由于调用SIGPIPE信号而导致管道后读取的循环在第一次迭代后终止,因为管道的LHS写入管道,其输出未被读取,因为在RHS上没有读取循环.以猫为例的例子不会退出,因为cat连续从输入读取,读取在读取一行后终止.

要了解此行为,请首先减少您的示例:

while : ; do pwd; done | { read -r line; echo $line; }
/Users/admin

所以读取在第一行之后终止.要验证此启用pipefail使用:

set -o pipefail

并检查退出状态:

while : ; do pwd; done | { read -r line; echo "$line"; }
echo $?
141

存在状态141是由于SIGPIPE.

要解决这个问题,现在改变你对while循环中管道的RHS的读取:

while : ; do pwd; sleep 5; done | { while read -r line; do echo "$line"; done; }
/Users/admin
/Users/admin
/Users/admin

现在你根本不会看到命令退出,因为read正在连续捕获来自LHS管道的所有数据.

dawei

【声明】:唐山站长网内容转载自互联网,其相关言论仅代表作者个人观点绝非权威,不代表本站立场。如您发现内容存在版权问题,请提交相关链接至邮箱:bqsm@foxmail.com,我们将及时予以处理。