Bash:为什么管道输入“读”只有当进入“同时阅读…”结构时工作?

我一直在尝试从程序输出读取环境变量的输入,如下所示:
echo first second | read A B ; echo $A-$B

结果是:

-

A和B总是空的。我阅读关于bash执行管道命令在子shell和基本上防止管道输入读取。但是,以下:

echo first second | while read A B ; do echo $A-$B ; done

似乎工作,结果是:

first-second

有人可以解释一下这里的逻辑是什么?是因为while … done构造中的命令实际上是在与echo相同的shell中执行的,而不是在子shell中执行的。

如何做一个循环对stdin和得到结果存储在一个变量

bash(和其他shell也),当你管道东西使用|到另一个命令,你将暗中创建一个fork,一个子shell,它是当前会话的子项,并且不能影响当前会话的环境。

所以这:

TOTAL=0
printf "%s %s\n" 9 4 3 1 77 2 25 12 226 664 |
  while read A B;do
      ((TOTAL+=A-B))
      printf "%3d - %3d = %4d -> TOTAL= %4d\n" $A $B $[A-B] $TOTAL
    done
echo final total: $TOTAL

不会给出预期的结果! :

9 -   4 =    5 -> TOTAL=    5
  3 -   1 =    2 -> TOTAL=    7
 77 -   2 =   75 -> TOTAL=   82
 25 -  12 =   13 -> TOTAL=   95
226 - 664 = -438 -> TOTAL= -343
echo final total: $TOTAL
final total: 0

其中计算的TOTAL不能在主脚本中重复使用。

倒转叉

通过使用bash过程替代,这里文档或这里字符串,你可以逆叉:

这里是字符串

read A B <<<"first second"
echo $A
first

echo $B
second

这里文件

while read A B;do
    echo $A-$B
    C=$A-$B
  done << eodoc
first second
third fourth
eodoc
first-second
third-fourth

外环:

echo : $C
: third-fourth

这里命令

TOTAL=0
while read A B;do
    ((TOTAL+=A-B))
    printf "%3d - %3d = %4d -> TOTAL= %4d\n" $A $B $[A-B] $TOTAL
  done < <(
    printf "%s %s\n" 9 4 3 1 77 2 25 12 226 664
)
  9 -   4 =    5 -> TOTAL=    5
  3 -   1 =    2 -> TOTAL=    7
 77 -   2 =   75 -> TOTAL=   82
 25 -  12 =   13 -> TOTAL=   95
226 - 664 = -438 -> TOTAL= -343

# and finally out of loop:
echo $TOTAL
-343

现在你可以在你的主脚本中使用$ TOTAL。

管道到命令列表

但是对于只对stdin工作,你可以在fork中创建一种脚本:

printf "%s %s\n" 9 4 3 1 77 2 25 12 226 664 | {
    TOTAL=0
    while read A B;do
        ((TOTAL+=A-B))
        printf "%3d - %3d = %4d -> TOTAL= %4d\n" $A $B $[A-B] $TOTAL
    done
    echo "Out of the loop total:" $TOTAL
  }

会给:

9 -   4 =    5 -> TOTAL=    5
  3 -   1 =    2 -> TOTAL=    7
 77 -   2 =   75 -> TOTAL=   82
 25 -  12 =   13 -> TOTAL=   95
226 - 664 = -438 -> TOTAL= -343
Out of the loop total: -343

注意:$ TOTAL不能在主脚本中使用(最后一个右花括号后)。

使用lastpipe bash选项

正如@CharlesDuffy正确地指出的,有一个bash选项用于改变这种行为。但为此,我们必须先禁用作业控制:

shopt -s lastpipe           # Set *lastpipe* option
set +m                      # disabling job control
TOTAL=0
printf "%s %s\n" 9 4 3 1 77 2 25 12 226 664 |
  while read A B;do
      ((TOTAL+=A-B))
      printf "%3d - %3d = %4d -> TOTAL= %4d\n" $A $B $[A-B] $TOTAL
    done

  9 -   4 =    5 -> TOTAL= -338
  3 -   1 =    2 -> TOTAL= -336
 77 -   2 =   75 -> TOTAL= -261
 25 -  12 =   13 -> TOTAL= -248
226 - 664 = -438 -> TOTAL= -686

echo final total: $TOTAL
-343

这将工作,但我(个人)不喜欢这,因为这不是标准,将不会帮助使脚本可读。另外,禁用作业控制对于访问此行为似乎很贵。

注意:作业控制认情况下仅在交互式会话中启用。因此,在正常脚本中不需要设置m。

因此,如果在控制台中运行或在脚本中运行,脚本中被遗忘的集合m将创建不同的行为。这将不会使这很容易理解或调试…

相关文章

用的openwrt路由器,家里宽带申请了动态公网ip,为了方便把2...
#!/bin/bashcommand1&command2&wait从Shell脚本并行...
1.先查出MAMP下面集成的PHP版本cd/Applications/MAMP/bin/ph...
1、先输入locale-a,查看一下现在已安装的语言2、若不存在如...
BashPerlTclsyntaxdiff1.进制数表示Languagebinaryoctalhexa...
正常安装了k8s后,使用kubect工具后接的命令不能直接tab补全...