通过stdout和stdin,父母和孩子之间的沟通

时间:2016-09-23 13:54:03

标签: c linux pipe fork exec

我在C中管道和分叉相对较新,

我有一个父进程和一个子进程,并尝试在它们之间进行通信。

我创建了一个管道,以便:父项的stdout将连接到子项的stdin,而子项的stdout将连接到父项的stdin。

我通过尝试让父母发出来测试这个" A"通过它的标准,然后孩子将收到" A"从它的stdin,之后它应该发回" B"从它的标准输出到父母的标准输入。

这是我的代码,

bool pget_input(void){
    char buffer[80];
    fprintf(stdout,"A");
    fprintf(stderr,"Parent sent out 'A' to child \n");
    do{
        if (!fgets(buffer, 20, stdin)) {
            fprintf(stderr,"Parent failed to recieve from child\n");
            return false;                                        
        }
        fprintf(stderr,"Parent receiving from child\n");
        if(strcmp(buffer, "B")==0){
            fprintf(stderr,"Parent success recieve child's reply");
        }
        continue;
    }while(true);
}
int main(int argc, char** argv){

    pid_t pid = fork();
    int pipe1[2];
    int pipe2[2];

    if (pipe(pipe1)==-1 || pipe(pipe2)==-1){
        fprintf(stderr,"pipe failed\n");
        exit(1);
    }

    if(pid > 0){
        //parent
        close(STDIN_FILENO);    //close stdin
        dup(pipe1[0]);          //replace stdin with pipe1 read
        close(STDOUT_FILENO);   //close stdout
        dup(pipe2[1]);          //replace stdout with pipe2 write
        //close pipe1 and pipe2 read write
        close(pipe1[0]);        
        close(pipe1[1]);
        close(pipe2[0]);
        close(pipe2[1]);
        pget_input();
        wait(0);
    }else if(pid ==0){
        //child
        close(STDOUT_FILENO);   //close stdout
        dup(pipe1[1]);          //replace stdout with pipe1 write
        close(STDIN_FILENO);    //close stdin
        dup(pipe2[0]);          //replace stdin with pipe2 read

        //close pipe1 and pipe2 read write
        close(pipe1[0]);        
        close(pipe1[1]);
        close(pipe2[0]);
        close(pipe2[1]);

        execl("./child", "not matter", NULL);
        perror("execvp failed");
           exit(1);

    }else{
        fprintf(stderr,"fork failed\n");
        exit(1);
    }
    fprintf(stderr,"Exiting\n");
    exit(1);

}

其中./child:

bool cget_input(void){
    char buffer[80], command[80];
    fprintf(stderr,"Child trying receive from parent\n");
    do{
        if (!fgets(buffer, 20, stdin)) {
            fprintf(stderr,"child failed to receive from parent\n");
            return false;                                        
        }
        sscanf(buffer, "%s", command);
        if(strcmp(buffer, "A") == 0){
            fprintf(stderr,"Child received 'A' from Parent and sending out 'B'\n");
            fprintf(stdout, "B");
        }
        continue;
    }while(true);
}
int main ()
{

    if(!cget_input()){
        return 0;
    }

    return 1;
}

结果,父母和孩子的stdin都没有收到任何东西, 没有通过

if (!fgets(buffer, 20, stdin)) 

有人可以指出我的代码出了什么问题吗? 谢谢,

1 个答案:

答案 0 :(得分:1)

您必须在之前设置管道

以下是需要完成的订单概述:

  1. 使用pipe()
  2. 创建管道
  3. 父母和孩子各自关闭管道的正确末端。
  4. 如果需要,父母和孩子可以重定向stdout,stdin用dup()和close()
  5. 此时,您可以在子进程中执行()一个新程序,该程序不需要知道已发生的重定向。
  6. 在fork之后,父级和子级是两个不同的进程,父级无权在此时修改子级的文件描述符。必须在fork之前设置任何共享文件描述符。