在 C 中使用管道在父子之间创建双向通信

Posted

技术标签:

【中文标题】在 C 中使用管道在父子之间创建双向通信【英文标题】:Create 2-way communication between parent and child using pipes in C 【发布时间】:2018-09-23 06:30:56 【问题描述】:

我正在尝试编写一个程序,允许进程与其子进程进行双向通信,即它可以向子进程发送消息,也可以从子进程接收消息。我第一次尝试创建 2 个管道并将管道的每一端链接到父子标准输入和标准输出:

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

int main(int argc, char** argv) 
    int firstPipe[2];
    int secondPipe[2];
    if (pipe(firstPipe) < 0) 
        perror("pipe");
        exit(1);
    
    if (pipe(secondPipe) < 0) 
        perror("pipe");
        exit(1);
    
    if (fork() != 0)  // child
        dup2(firstPipe[0], 0);
        dup2(secondPipe[1], 1);
        char input[70];
        printf("Mr. Stark...");
        fgets(input, 70, stdin);
        fprintf(stderr, "%s I don't wanna go...\n", input);
     else  // parent
        dup2(secondPipe[0], 0);
        dup2(firstPipe[1], 1);
        char message[70];
        fgets(message, 70, stdin);
        printf("%s I don't feel so good...", message);
    
    return 0;

这个程序应该是从孩子发消息给家长,然后家长给孩子发回复,然后孩子打印最终结果(Stark先生...我感觉不太好...我不想去...)到stderr,但它不起作用:(当我尝试运行它时,它会冻结,好像其中一个进程(或两个进程)正在等待输入。是我的代码有问题吗?我也愿意接受其他方法的建议,只要最终结果有效。感谢您的帮助。

【问题讨论】:

【参考方案1】:

fgets 一直读取直到看到换行符(或缓冲区已满)。

父级以

开头
fgets(message, 70, stdin);

等待排队。

子输出

printf("Mr. Stark...");

然后也等待:

fgets(input, 70, stdin);

"Mr. Stark..." 不包含换行符。事实上,它可能根本没有被发送,而是在stdout 内部缓冲,但这可以在printf 之后使用fflush(stdout) 修复。

但即便如此,fgets 仍会等待一个永远不会出现的换行符。

修复:

printf("Mr. Stark...\n");
fflush(stdout);

【讨论】:

谢谢。这部分解决了我的问题。它打印出不包含I don't feel so good 部分的Mr Stark... I don't wanna go...。我已经添加了换行符和 fflush 但它似乎没有被修复。我错过了什么吗? @n.m 是的,我注意到字符串中间有一个换行符,这似乎是问题所在,但还有什么办法可以解决这个问题吗? @MikePham fgets 不会删除换行符。您可能想自己删除它。

以上是关于在 C 中使用管道在父子之间创建双向通信的主要内容,如果未能解决你的问题,请参考以下文章

管道和FIFO

管道双向通信

进程间通信的方式

进程间通信

进程间通信

需要检查管道是不是双向工作