c - 如何与另一个外部程序进行通信

标签 c linux stdio

我正在尝试从外部程序写入 stdin 并从 stdout (和 stderr )读取,而不更改代码。

我尝试过使用命名管道,但 stdout 在程序终止之前不会显示,并且 stdin 仅适用于第一个输入(然后 cin 为 null )。

我尝试过使用/proc/[pid]/fd 但它只能从终端而不是程序进行写入和读取。

我尝试为此编写一个字符设备文件并且它有效,但一次只能运行一个程序(这需要一次适用于多个程序)。

目前,据我所知,我可以编写能够在多个程序之间复用 io 的驱动程序,但我认为这不是“正确”的解决方案。

其主要目的是通过网络界面查看程序的提要。我确信必须有某种方法来做到这一点。有什么是我以前没有尝试过的吗?

最佳答案

执行此操作的典型方法是:

  1. 使用pipe(2)创建匿名管道(不是命名管道)新进程的标准流的系统调用
  2. 调用 fork(2)生成子进程
  3. close(2)父级和子级中管道的相应端(例如,对于 stdin 管道,关闭父级中的读取端并关闭子级中的写入端;对于 stdout 和 stderr 管道反之亦然)
  4. 使用dup2(2)在子级中将管道文件描述符复制到文件描述符 0、1 和 2 上,然后 close(2) 剩余的旧描述符
  5. exec(3)子进程中的外部应用程序
  6. 在父进程中,同时写入子进程的 stdin 管道并从子进程的 stdout 和 stderr 管道读取。然而,根据 child 的行为方式,如果你不小心,这很容易导致僵局。避免死锁的一种方法是生成单独的线程来处理 3 个流中的每一个;另一种方法是使用 select(2)系统调用等待直到可以无阻塞地读取/写入其中一个流,然后处理该流。

即使您正确执行了这一切,您可能仍然无法立即看到程序的输出。这通常是由于缓冲标准输出造成的。通常,当 stdout 进入终端时,它是行缓冲的——在写入每个换行符后它会被刷新。但是,当 stdout 是管道(或任何其他不是终端的东西,如文件或套接字)时,它是完全缓冲的,并且只有在程序输出完整缓冲区的数据(例如 4 KB)时才会写入。

许多程序都有命令行选项来更改其缓冲行为。例如,grep(1) 具有 --line-buffered 标志,以强制它对输出进行行缓冲,即使 stdout 不是终端也是如此。如果您的外部程序有这样的选项,您可能应该使用它。如果没有,仍然可以更改缓冲行为,但您必须使用一些偷偷摸摸的技巧 - 请参阅 this questionthis question了解如何做到这一点。

关于c - 如何与另一个外部程序进行通信,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15516020/

相关文章:

Linux 中的 Python shell

windows - LIBDBUS 菜单-GLIB-警告

c - 将输出流式传输到 stdout,就地状态更新到 stderr

c - fscanf 始终返回 0(已关闭)

c - 如何编写通用函数来获取原子列表的长度

C 代码卡在 scanf() 语句

c - C 语言编程中的 extern

linux - 我的 nuxt 应用程序在 CodeSandbox 上运行良好,但在我的本地机器上运行不正常

c - 当子进程执行带有 setuid 位设置的 execve 时,fork() 中的文件描述符

清除换行符后的输入行