将数据从父级传递给子级并使用Unix C传回

时间:2013-09-10 21:22:37

标签: c unix parent pipe

我是C的新手,我正在尝试学习如何在Unix环境中使用管道将数据从父进程传输到子进程,反之亦然。在下面的代码中,我收到一个命令行参数,并根据参数的值构建一个char数组。然后我使用管道将char数组传递给子节点,子节点将执行一个名为vc的程序。该程序根据char数组返回一个数字结果。我的问题是如何使用第二个管道将结果返回给父级?此外,一旦父母拥有它,我怎么能将它打印到屏幕,因为父设置将输出发送给孩子?谢谢。

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/wait.h>

int main(int argc,char *argv[])
{
    int 
        pfildes[2],
        pid,
        argNumber;
    char
        buffer[256],
        charString[1024]; 
    //Check for arguments 
    if(argc != 2) {
        printf("No command line arguements given.\n");
        argNumber=10; //default
    }
    else
       argNumber=atoi(argv[1]);

    //***********************************
    //Build charString based on argNumber
    //***********************************       

    //create pipes
    if(pipe(pfildes) == -1) {
        //error occured when creating pipe
        perror("demo");
        exit(1);
    }
    //create child process
    if((pid=fork()) < 0) {
        //error occured when forking child
        perror("demo");
        exit(1);
    }
    else if(pid > 0) {
        //parent process
        close(pfildes[0]);
        dup2(pfildes[1],1);
        printf("%s", charString);

        close(pfildes[1]);

        perror("demo");
        _exit(1);
    }
    else {
        //child process
        close(pfildes[1]);
        dup2(pfildes[0],0);
        execlp("/usr/bin/vc","vc", NULL);
        close(pfildes[0]);

        perror("demo");
        exit(1);
    }

    while(wait(NULL) >0);
    return 0;
}

1 个答案:

答案 0 :(得分:1)

您可以使用socketpair()代替pipe()来生成父进程和子进程之间的双向通信通道:

//...
if (socketpair(PF_UNIX, SOCK_STREAM, 0, pfildes) == -1) {
    //error occured when socket pair
    perror("demo: socketpair");
    exit(1);
}
//..

在子进程中,您可以在调用dup()之前将exec()对中的一个放入输入和输出中:

//...
else {
    //child process
    close(pfildes[1]);
    dup2(pfildes[0],0);
    dup2(pfildes[0],1);
    dup2(pfildes[0],2);
    close(pfildes[0]);
    execlp("/usr/bin/vc","vc", NULL);
    perror("demo: child exec");
    exit(1);
}
//...

在父进程中,您可以使用fdopen()从文件描述符创建FILE *,因此您不需要dup()覆盖现有的stdout文件描述符:

//...
else if(pid > 0) {
    //parent process
    close(pfildes[0]);
    FILE *to_child = fdopen(dup(pfildes[1]), "w");
    FILE *from_child = fdopen(dup(pfildes[1]), "r");
    close(pfildes[1]);
    fprintf(to_child, "%s", charString);
    while (fgets(buf, sizeof(buf), from_child) != NULL) {
        //...do something with output
    }
    //...
} else { //...