管道输出和Bash中的捕获退出状态怎么弄?

内容来源于 Stack Overflow,并遵循CC BY-SA 3.0许可协议进行翻译与使用

  • 回答 (10)
  • 关注 (0)
  • 查看 (272)

我想在Bash中执行一个长时间运行的命令,并且都捕获它的退出状态

所以我这么做:

command | tee out.txt
ST=$?

问题在于,变量ST捕获了tee而不是命令。我怎么解决这个问题?

提问于
用户回答回答于

所以我想给出一个像Lesmana一样的答案,但我认为我的答案可能更简单,也可能更有利一些,纯粹的Bourne-shell解决方案:

# You want to pipe command1 through command2:
exec 4>&1
exitstatus=`{ { command1; printf $? 1>&3; } | command2 1>&4; } 3>&1`
# $exitstatus now has command1's exit status.

我认为这最好从内部解释--命令1将在stdout(文件描述符1)上执行并打印其常规输出,然后,printf将在其stdout上执行并打印icommand 1的退出代码,但该stdout将被重定向到文件描述符3。

当命令1运行时,它的stdout被管道传输到命令2(printf的输出永远不会到达命令2,因为我们将它发送到文件描述符3,而不是管道读取的1)。然后,我们将命令2的输出重定向到文件描述符4,这样它也不会出现在文件描述符1中--因为我们希望文件描述符1在稍后的时间内是免费的。因为我们将把文件描述符3上的printf输出返回到文件描述符1中

最后一点魔法是exec 4>&1我们作为一个单独的命令进行了操作--它将文件描述符4作为外部shell的stdout的副本打开。命令替换将从标准输出命令的角度捕获任何内容,它实际上仍然会转到脚本的整体文件描述符1。

(()exec 4>&1必须是一个单独的命令,因为许多常见的shell都不喜欢在尝试写入命令替换内的文件描述符时,该命令是在使用替换的“外部”命令中打开的。因此,这是最简单的便携方式。)

你可以以一种不那么技术性和更有趣的方式来看待它,就好像命令的输出是相互跳跃的:命令1管道到命令2,然后printf的输出跳过命令2,这样命令2就不会捕捉到它。然后命令2的输出跳过命令替换,就像printf正好及时地被替换所捕获,从而使它在变量中结束,命令2的输出继续以其愉快的方式被写入标准输出。就像在普通管道里一样。

而且,据我所知,$?仍然将包含管道中第二个命令的返回代码,因为变量赋值、命令替换和复合命令对它们中的命令的返回代码都是有效透明的,因此命令2的返回状态应该被传播出去--这就是为什么我认为这可能是一个比Lesmana提出的更好的解决方案。

根据Lesmana提到的警告,命令1可能在某个时候使用文件描述符3或4,因此你可以这样做:

exec 4>&1
exitstatus=`{ { command1 3>&-; printf $? 1>&3; } 4>&- | command2 1>&4; } 3>&1`
exec 4>&-

命令从启动它们的过程继承文件描述符,因此整个第二行将继承文件描述符4,复合命令后面跟着3>&1将继承文件描述符3。所以4>&-确保内部复合命令不会继承文件描述符4,并且3>&-不会继承文件描述符3,所以命令1获得了一个“更干净”、更标准的环境。你也可以移动内部4>&-旁边3>&-,但不尽量限制它的范围。

热门问答

mqtt通信数据转发至另一Topic失败?

推荐已采纳

设置转发的这个产品的数据格式设置的是自定义,系统会认为所有数据都是二进制数据。如果需要使用json解析,需要设置产品数据格式是JSON。

腾讯云GPU服务器不能联外网吗?

小爱同学

腾讯云 · 技术支持 (已认证)

推荐
腾讯云GPU服务器可连外网,GPU 云服务器提供和标准CVM 云服务器一致的方便快捷的管理方式。 图片.png GPU云服务器作为CVM云服务器的一类特殊实例,购买、 操作、维护等方式与CVM云服务器一致 图片.png GPU 云服务器(GPU Cloud Computin...... 展开详请

win服务器怎么给文件夹配置755权限?

推荐
下面以腾讯云win服务器(Windows Server 2016 数据中心版 64位中文版)为文件夹配置755权限为例 1.右击【属性】 图片.png 2 .选择【安全】- 【编辑】 图片.png 3. 可对当前文件进行755权限配置 图片.png 要修改某个文件的权...... 展开详请

腾讯云sdk 兼容JDK6?

推荐

如果你说的是https://cloud.tencent.com/document/sdk/Java的话,jdk最低版本是1.7,不支持1.6

android 离线推送 为什么setOfflinePushListener不回调?

嗨喽你好摩羯座
推荐
您好,使用云通信 IM SDK 的通知栏提醒,建议参考:https://cloud.tencent.com/document/product/269/9234 中的描述来操作,通知栏提醒的内容由类 TIMOfflinePushNotification 来定义,可以通过这个类对外...... 展开详请

为什么cmq的topic配置订阅者为queue,向topic发送消息无法到达queue?

是的, 向topic发送消息应该会立即投递到订阅者。您可以检查您配置的队列名称是否正确且是真实存在的队列。如还不能解决您的问题,您可以点击控制台右上角的“工单”,进行问题进一步的排查,腾讯云会有专业的售后24小时为您服务。

所属标签

扫码关注云+社区