- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在编写一个库,它应该在子进程中执行程序,捕获输出,并以逐行(字符串 vector )的方式提供输出。有一个用于 STDOUT 的 vector ,一个用于 STDERR,还有一个用于“STDCOMBINED”,即所有输出均按照程序打印的顺序排列。子进程通过两个管道连接到父进程。一根管道用于 STDOUT,一根管道用于 STDERR。在父进程中我从管道的读取端读取,在子进程中我 dup2()
'ed STDOUT/STDERR 到管道的写入端。
我的问题:我想捕获 STDOUT、STDERR、和“STDCOMBINED”(=均按照它们出现的顺序)。但组合 vector 中的顺序与原始顺序不同。
我的方法:我迭代直到两个管道都显示 EOF 并且子进程退出。在每次迭代中,我从 STDOUT 中读取一行(或 EOF),从 STDERR 中读取一行(或 EOF)。到目前为止这有效。但是,当我捕获父进程中出现的行时,STDOUT 和 STDERR 的顺序并不相同,就好像我在 shell 中执行程序并查看输出一样。
为什么会这样?我该如何解决这个问题?这有可能吗?我知道在子进程中我可以将 STDOUT 和 STDERR 都重定向到单个管道,但我需要分别使用 STDOUT 和 STDERR 以及“STDCOMBINED”。
PS:我熟悉libc/unix系统调用,例如dup2()
, pipe()
等等。因此我没有发布代码。我的问题是关于一般方法,而不是特定语言的编码问题。我在 Rust 中针对原始 libc 绑定(bind)执行此操作。
PPS:我制作了一个简单的测试程序,其中混合了 5 个 stdout 和 5 个 stderr 消息。这足以重现问题。
最佳答案
At each iteration I read exactly one line (or EOF) from STDOUT and exactly one line (or EOF) from STDERR.
这就是问题所在。仅当这正是子进程中的输出顺序时,才会捕获正确的顺序。
您需要捕捉野兽的异步本质:使管道端点非阻塞,在管道上select
*,并在select
后立即读取存在的任何数据> 返回。然后您将捕获输出的正确顺序。当然,现在你不能读取“恰好一行”:你必须读取任何可用的数据,这样你就不会阻塞,并维护一个每个管道的缓冲区,在其中附加新数据,提取存在的任何行,将未处理的输出推到开头,然后重复。您还可以使用循环缓冲区来保存一点 memcpy
-ing,但这可能不是很重要。
既然你在 Rust 中这样做,我想你已经有了一个可以利用的良好的异步 react 模式(我猜我被 go 宠坏了,并将希望寄托在毫无戒心的人身上)。
*始终更喜欢特定于平台的高性能原语,例如 epoll
on Linux , /dev/poll
on Solaris , pollset
&c. on AIX
另一种可能性是使用LD_PRELOAD
启动目标进程,并使用一个专用库来接管glibc的POSIX write
,检测对管道的写入,并封装此类写入(并且只有那些)在数据包中,通过在其前面添加一个 header ,该 header 中存储有(自动更新的)进程范围递增计数器以及写入的大小。此类 header 可以在管道的另一端轻松解码,以便以更高的成功机会重新排序写入。
关于从子进程中单独并同时捕获 stdout/stderr 会导致总顺序错误 (libc/unix),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/65053552/
我有以下代码来捕获和处理运行命令输出。 如何修改它以便运行命令窗口显示输出并同时记录输出?更换 @SW_HIDE与 @SW_SHOW (或等效的)只显示一个空白的命令窗口。 类似于linux的东西te
[已编辑] 用于处理 subprocess.Popen 标准输出的 2 个选项是 stdout="a_file_name" 和 stdout=subprocess.PIPE。 stderr 可以通过
c99 的 7.19.3/7 节指出: At program start-up, three text streams are predefined and need not be opened ex
我正在运行以下 proc_open 函数。加载页面时,出现错误: Use of undefined constant STDOUT - assumed 'STDOUT'` 如何正确设置STDOUT和S
我有一个运行多个进程的开发堆栈:网络服务器、自动测试、后台编译等。所有这些都是基本的命令行命令,例如 node app.js 或 lein midje :autotest. 是否可以使用一个脚本在“后
我正在使用 SLURM 在 super 计算机上调度作业。我已设置 --output=log.out 选项,将作业标准输出中的内容放入文件 (log.out)。我发现该文件每 30-60 分钟更新一次
ansible/ansible-playbook 版本:2.1.2.0/2.2.0.0 我正在尝试使用 yum/apt 安装软件包,但由于安装软件包的存储库位于 packagecloud.io 中,有
bala@hp:~$ echo "Hello World" > stdout bala@hp:~$ cat stdout Hello World bala@hp:~$ echo "Hello Worl
在从 Fortran 编写的外部库中调用嘈杂的函数之前,我正在使用如下代码重定向标准输出: // copy standard output out = dup(STDOUT_FILENO); // c
这个问题在这里已经有了答案: How can I pipe stderr, and not stdout? (11 个答案) 关闭 6 年前。 我有一个程序,我想检查其 STDERR 输出并在其上运
我正在从 perl 运行一个 java 应用程序 这是脚本 #!/usr/bin/perl use strict; use warnings; $| = 1; my $args = join (" "
我正在尝试将 python 脚本的 STDOUT 重定向到一个文件。 如果 STDOUT 是从 sys 导入的,脚本的输出不会被重定向到一个文件: from sys import stdout std
我正在尝试使用 PHP 和 Apache(在 Docker 的前台运行)写入 stdout(或 stderr)。 这些作品: file_put_contents( "php://stderr","wo
我正在尝试重定向标准输出,以便 Windows 应用程序中的 printf 将转到我选择的文件。 我这样做: outFile = fopen("log.txt", "w"); *stdout = *o
在 Ruby 中,$stdout(前面有一个美元符号)和 STDOUT(全部大写)有什么区别?在进行输出重定向时,应该使用哪个,为什么? $stderr 和 STDERR 也是如此。 编辑: 刚找到一
让我们看看这个Hello World程序 #include int main(int argc, char ** argv) { printf("Hello, World!"); c
我在 64 位机器上运行 Ubuntu 20.04。 我想将 stdout 重定向到从 memfd_create 获得的描述符。似乎使用 memfd_create 创建的匿名文件只有在 stdout
我想在 Python 脚本末尾捕获控制台输出。也就是说,我既想正常打印到控制台,又想在执行结束时将控制台输出保存到文件中。 我看过各种相关的 SO 问题 1 , 2 , 3尽管他们要么简单地重定向输出
我知道可以将两者都重定向到特定文件: ./command 1> out.log 2> err.log 或 ./command 1>test.log 2>&1 将两者写入文件。但是我不知道在只打印其中一
我知道可以将两者都重定向到特定文件: ./command 1> out.log 2> err.log 或 ./command 1>test.log 2>&1 将两者写入文件。但是我不知道在只打印其中一
我是一名优秀的程序员,十分优秀!