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 %sn" 9 4 3 1 77 2 25 12 226 664 | while read A B;do ((TOTAL+=A-B)) printf "%3d - %3d = %4d -> TOTAL= %4dn" $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= %4dn" $A $B $[A-B] $TOTAL done < <( printf "%s %sn" 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 %sn" 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= %4dn" $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 %sn" 9 4 3 1 77 2 25 12 226 664 | while read A B;do ((TOTAL+=A-B)) printf "%3d - %3d = %4d -> TOTAL= %4dn" $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将创建不同的行为。这将不会使这很容易理解或调试… (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |