为什么读取 StdoutPipe 会导致输出乱序?
来源:stackoverflow
时间:2024-02-19 20:45:25 164浏览 收藏
从现在开始,努力学习吧!本文《为什么读取 StdoutPipe 会导致输出乱序?》主要讲解了等等相关知识点,我会在golang学习网中持续更新相关的系列文章,欢迎大家关注并积极留言建议。下面就先一起来看一下本篇正文内容吧,希望能帮到你!
我有以下函数,该函数调用我无法在此处发布的命令。以前我使用 cmd.stdout = os.stdout
在命令行上获取命令输出,如下所示,效果很好:
func run(cmd *exec.cmd) { cmd.stdout = os.stdout if err := cmd.start(); err != nil { panic(err) } if err := cmd.wait(); err != nil { panic(err) } return }
这给了我以下命令行输出,这很棒:
[loader] loading programm... [programm] foo bar
现在我需要先与输出交互,所以我尝试使用 stdoutpipe
:
func run(cmd *exec.cmd) { stdout, err := cmd.stdoutpipe() if err != nil { panic(err) } if err := cmd.start(); err != nil { panic(err) } reader := bufio.newreader(stdout) for { s, err := reader.readstring('\n') if err != nil { break } // do some evaluation of s here fmt.print(s) } if err := cmd.wait(); err != nil { panic(err) } return }
但这会打乱消息的顺序,我没有得到以下输出:
[programm] foo bar [loader] loading programm...
为什么输出不再按顺序排列?如何在使用 stdoutpipe 时保持输出按顺序排列?
编辑:
如果我尝试像这样使用 io.multiwriter
,也会发生同样的情况:
func run(cmd *exec.Cmd) { var reader bytes.Buffer cmd.Stdout = io.MultiWriter(os.Stdout, &reader) if err := cmd.Start(); err != nil { panic(err) } for { s, err := reader.ReadString('\n') if err != nil { break } fmt.Print(s) } if err := cmd.Wait(); err != nil { panic(err) } return }
解决方案
很大程度上取决于您运行的程序。
我使用 stdbuf -o0
来停用缓冲
这仅影响程序的 stdout
(POSIX 系统上的文件描述符 1),而不影响其 stderr
(文件描述符 2)。但 POSIX 系统通常使用以下默认值,至少对于 C 语言程序或使用 C 运行时系统的程序来说是这样:
- stdin:缓冲并不是特别相关,但输入通常会完全缓冲。
- stdout:如果文件描述符 1 是“tty”,则行缓冲;否则完全缓冲。
- stderr:无缓冲。
确定什么是“tty”是通过调用C库isatty
函数来完成的,该函数通常使用termios操作来完成此操作。例如,glibc isatty
调用 TCGETATTR
并验证它是否返回且没有错误。 glibc tcgetattr 使用 TCGETATTR
ioctl。因此“tty 设备”是实现 TCGETATTR
的设备。
过去使用的实际串行线路(例如 COM1 和 COM2),或者过去使用的终端集中器设备(例如 DH/DM 或 DZ),当然要实现 TCGETATTR
。伪终端也是如此,例如交互式 ssh
会话、script
和其他程序所使用的伪终端。这使得 1970 年代和 1980 年代的所有旧软件能够继续与虚拟化 ASR33、ADM3A 或 DECwriter 一起工作。1
无论如何,这里的关键是管道(或套接字或类似的东西)不实现 TCGETATTR
,因此如果 stdout 是管道,则 isatty(1)
为 0(假)。这使得标准输出完全缓冲。
请记住,这一切都发生在您运行的程序中。您自己的程序读取其他程序的输出(不一定)不是问题。您的程序仅在其他程序实际发送输出时才获取输出。在完全缓冲模式下,其他程序仅在填满缓冲区块或调用某些输出刷新操作时发送输出。
一些程序将输出混合到 stdout - 他们通常假设是行缓冲的,因为编写(有缺陷的)程序的人仅使用 tty 输出对其进行测试 - 并输出到 stderr。当标准输出重定向到文件或管道时,这些程序的行为很奇怪。像 stdbuf
这样的程序(可以作为库预加载实现)和 hence might not work on setuid programs 可以用来解决这些问题。
在 Go 中,缓冲更多地取决于您自己的程序。 Go 不使用 POSIX 设置。例如,请参阅 Idiomatically buffer os.Stdout。
1我更喜欢我的虚拟 Tek4025、Heathkit H19 等,尽管对于旧的打印终端有一些话要说!或者是旧的 AAA 终端,在斜视模式下最多可显示 60 行。 OK,Terminal、iterm2 等已经打败它们了。
文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《为什么读取 StdoutPipe 会导致输出乱序?》文章吧,也可关注golang学习网公众号了解相关技术文章。
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习