我正在尝试从外部程序写入 stdin 并从 stdout (和 stderr )读取,而不更改代码。
我尝试过使用命名管道,但 stdout 在程序终止之前不会显示,并且 stdin 仅适用于第一个输入(然后 cin 为 null )。
我尝试过使用/proc/[pid]/fd 但它只能从终端而不是程序进行写入和读取。
我尝试为此编写一个字符设备文件并且它有效,但一次只能运行一个程序(这需要一次适用于多个程序)。
目前,据我所知,我可以编写能够在多个程序之间复用 io 的驱动程序,但我认为这不是“正确”的解决方案。
其主要目的是通过网络界面查看程序的提要。我确信必须有某种方法来做到这一点。有什么是我以前没有尝试过的吗?
最佳答案
执行此操作的典型方法是:
- 使用
pipe(2)
创建匿名管道(不是命名管道)新进程的标准流的系统调用 - 调用
fork(2)
生成子进程 -
close(2)
父级和子级中管道的相应端(例如,对于 stdin 管道,关闭父级中的读取端并关闭子级中的写入端;对于 stdout 和 stderr 管道反之亦然) - 使用
dup2(2)
在子级中将管道文件描述符复制到文件描述符 0、1 和 2 上,然后close(2)
剩余的旧描述符 -
exec(3)
子进程中的外部应用程序 - 在父进程中,同时写入子进程的 stdin 管道并从子进程的 stdout 和 stderr 管道读取。然而,根据 child 的行为方式,如果你不小心,这很容易导致僵局。避免死锁的一种方法是生成单独的线程来处理 3 个流中的每一个;另一种方法是使用
select(2)
系统调用等待直到可以无阻塞地读取/写入其中一个流,然后处理该流。
即使您正确执行了这一切,您可能仍然无法立即看到程序的输出。这通常是由于缓冲标准输出造成的。通常,当 stdout 进入终端时,它是行缓冲的——在写入每个换行符后它会被刷新。但是,当 stdout 是管道(或任何其他不是终端的东西,如文件或套接字)时,它是完全缓冲的,并且只有在程序输出完整缓冲区的数据(例如 4 KB)时才会写入。
许多程序都有命令行选项来更改其缓冲行为。例如,grep(1)
具有 --line-buffered
标志,以强制它对输出进行行缓冲,即使 stdout 不是终端也是如此。如果您的外部程序有这样的选项,您可能应该使用它。如果没有,仍然可以更改缓冲行为,但您必须使用一些偷偷摸摸的技巧 - 请参阅 this question和 this question了解如何做到这一点。
关于c - 如何与另一个外部程序进行通信,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15516020/