如何使用popen?

Posted

技术标签:

【中文标题】如何使用popen?【英文标题】:How to use popen? 【发布时间】:2015-05-15 12:03:47 【问题描述】:

我正在尝试与stdinstdout 进行进程间通信。我找到的 Posix 函数是 popen,但我没有编写一个有效的示例代码。请帮我完成这项工作。

我必须使用dup 吗?我可以看到一些 Google 使用它的例子。但是dup 的 Linux 手册确实不能帮助我理解如何使用它。

交流

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int main(void)
    char *s;

    for(;;)
        scanf("%ms",&s);
        printf("%s\n",s);
        if(!strcmp(s,"quit"))
            free(s);
            printf("bye~\n");
            exit(EXIT_SUCCESS);
        
        free(s);
    

b.c

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int main(void)
    FILE *fRead;
    FILE *fWrite;
    char *s;
    int i;

    fRead=popen("~/a.out","r");
    fWrite=popen("~/a.out","w");
    for(i=1;i<=10;++i)
        fprintf(fWrite,"%d",i);
        fscanf(fRead,"%ms",&s);
        printf("%s\n",s);
        free(s);
    

【问题讨论】:

你知道你开始a.out 两次,不是吗? @alk 我确实看到了问题,那么如何同时读取输出? 使用popen("~/a.out &gt;output", "r") 并阅读output @xiver77,或使用pipe()/dup()/fork()/exec() 使用覆盖的IO 启动进程,然后使用poll() 多路复用输入/输出。 【参考方案1】:

按照 POSIX 的定义,管道是一种单向通信机制——它们只在一个方向上工作。为了同时重定向标准输入和标准输出,您需要创建两个管道——而popen 函数不能这样做。

虽然不太方便,但直接使用系统调用forkpipedup2exec,实现你想要的并不难:

rc = pipe(p1);
if(rc < 0) ...
rc = pipe(p2);
if(rc < 0) ...

rc = fork();
if(rc < 0) 
    ...
 else if(rc == 0) 
    /* child */
    close(p1[0]);
    close(p2[1]);
    dup2(p1[1], 1);
    dup2(p2[0], 0);
    execlp(...);
    exit(1);
 else 
    /* parent */
    close(p1[1]);
    close(p2[0]);
    ...

还有其他解决方案——您可以使用socketpair 系统调用来避免需要两个管道,甚至可以直接使用 Unix 域套接字。

【讨论】:

以上是关于如何使用popen?的主要内容,如果未能解决你的问题,请参考以下文章

如何使用子进程 Popen.communicate() 方法?

如何从 subprocess.Popen 使用 STDIN [重复]

如何将 Subprocess.popen() 与 pyinstaller 一起使用?

如何在 Windows 上使用带有内置命令的 subprocess.Popen

子进程 Popen 和 call 有啥区别(我该如何使用它们)?

如何使用 Popen 同时写入标准输出和日志文件?