C ++输入和输出管道都连接到外部程序

时间:2018-08-24 03:15:42

标签: c++ process pipe system

我试图用一些输入来调用外部程序,并在程序中从中检索输出。

看起来像;

  

(一些输入)| (外部程序)| (检索输出)

我首先考虑使用popen(),但由于管道不是双向的,似乎无法实现。

有什么简单的方法可以在 linux 中处理此类问题?

我可以尝试制作一个临时文件,但是如果可以在不访问磁盘的情况下清楚地处理它,那就太好了。

有解决方案吗?谢谢。

1 个答案:

答案 0 :(得分:4)

在Linux上,您可以使用pipe函数:打开两个新管道,每个方向一个,然后使用fork创建一个子进程,此后,您通常会关闭未使用的文件描述符(请阅读在父级上结束,在要发送给父级的子级上在管道的子级上写完,反之亦然(在另一个管道上),然后使用execve或其前端之一启动应用程序。

如果您将管道的文件描述符dup2到标准控制台文件句柄(STDIN_FILENO / STDOUT_FILENO;每个进程分别),则应该能够使用std::cin / std::cout与其他进程进行通信(您可能只想对孩子进行通信,因为您可能希望将控制台保留在父目录中)。不过,我尚未对此进行测试,因此留给您。

完成后,您wait or waitpid仍要终止子进程。可能类似于以下代码:

int pipeP2C[2], pipeC2P[2];
// (names: short for pipe for X (writing) to Y with P == parent, C == child)

if(pipe(pipeP2C) != 0 || pipe(pipeC2P) != 0)
{
    // error
    // TODO: appropriate handling
}
else
{
    int pid = fork();
    if(pid < 0)
    {
        // error
        // TODO: appropriate handling
    }
    else if(pid > 0)
    {
        // parent
        // close unused ends:
        close(pipeP2C[0]); // read end
        close(pipeC2P[1]); // write end

        // use pipes to communicate with child...

        int status;
        waitpid(pid, &status, 0);

        // cleanup or do whatever you want to do afterwards...
    }
    else
    {
        // child
        close(pipeP2C[1]); // write end
        close(pipeC2P[0]); // read end
        dup2(pipeP2C[0], STDIN_FILENO);
        dup2(pipeC2P[1], STDOUT_FILENO);
        // you should be able now to close the two remaining
        // pipe file desciptors as well as you dup'ed them already
        // (confirmed that it is working)
        close(pipeP2C[0]);
        close(pipeC2P[1]);

        execve(/*...*/); // won't return - but you should now be able to
                         // use stdin/stdout to communicate with parent
    }
}