在C中实现管道

时间:2014-02-20 17:01:35

标签: c linux pipe

我正在尝试在C中实现管道,例如 - $ ls | wc | wc

我写了以下代码 -

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

void run_cmd(char *cmd, int* fd_in, int* fd_out)
{
    int c = fork();

    if (c==0)
    {
        if (fd_in != NULL)
        {
            close(fd_in[1]);
            dup2(fd_in[0], 0);
        }
        if (fd_out != NULL)
        {
            close(fd_out[0]);
            dup2(fd_out[1],1);
        }
        execlp(cmd, cmd, NULL);
    }
}

int main(int argc, char **argv)
{
    int fd_1[2], fd_2[2], i;
    pipe(fd_1);
    pipe(fd_2);

    run_cmd(argv[1], NULL, fd_1);

    for( i=2; i<argc-1; i++)
    {
        if (i%2 == 0)
            run_cmd(argv[i], fd_1, fd_2);
        else
            run_cmd(argv[i], fd_2, fd_1);
    }
    if (i%2 == 0)
        run_cmd(argv[i], fd_1, NULL);
    else
        run_cmd(argv[i], fd_2, NULL);
}

这适用于两个参数,例如 - $./a.out ls wc

但是当我尝试使用两个以上的参数时它不起作用。

有人请告诉我我的代码有什么问题,或者其他任何方式吗?

2 个答案:

答案 0 :(得分:15)

这几乎没有错误检查,但为什么这么复杂?

int main (int argc, char ** argv) {
    int i;

    for( i=1; i<argc-1; i++)
    {
        int pd[2];
        pipe(pd);

        if (!fork()) {
            dup2(pd[1], 1); // remap output back to parent
            execlp(argv[i], argv[i], NULL);
            perror("exec");
            abort();
        }

        // remap output from previous child to input
        dup2(pd[0], 0);
        close(pd[1]);
    }

    execlp(argv[i], argv[i], NULL);
    perror("exec");
    abort();
}

答案 1 :(得分:0)

如果您仍然对您的来源无效的原因感兴趣(谢尔盖的解决方案无论如何都更好):

问题不在于关闭父进程中fd_1的写入端。因此,argv[1]和父亲都是该管道的作者,这引起了混乱。请不要询问更多详细信息(特别是如果只使用一个管道,为什么不会发生概率)但如果您在第一次调用{close( fd_1[1] );之后添加run_cmd(),则原始源将使用树进程运行{1}}

相关问题