使用dup2进行管道传输

20

我该如何使用dup2来执行以下命令?

ls -al | grep alpha | more
2个回答

32

这是一个简单的示例,使用了前两个命令。您需要使用pipe()函数创建一个管道,在ls和grep之间以及在grep和more之间建立一个管道。dup2所做的是将文件描述符复制到另一个文件描述符中。管道通过将fd[0]中的输入连接到fd[1]的输出来工作。您应该阅读pipe和dup2的手册页。如果您有其他疑问,我可能会尝试简化示例。

#include <sys/types.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>

#define READ_END 0
#define WRITE_END 1

int 
main(int argc, char* argv[]) 
{
    pid_t pid;
    int fd[2];

    pipe(fd);
    pid = fork();

    if(pid==0)
    {
        printf("i'm the child used for ls \n");
        dup2(fd[WRITE_END], STDOUT_FILENO);
        close(fd[WRITE_END]);
        execlp("ls", "ls", "-al", NULL);
    }
    else
    { 
        pid=fork();

        if(pid==0)
        {
            printf("i'm in the second child, which will be used to run grep\n");
            dup2(fd[READ_END], STDIN_FILENO);
            close(fd[READ_END]);
            execlp("grep", "grep", "alpha",NULL);
        }
    }

    return 0;
}

可以通过只使用一个fork()并将原始进程用于ls或grep来改进它,但是我会把这个留给你:P - theprole
我们应该如何将 grep 命令的输出传送给 more 呢?这只是简单地将两个进程连接起来吗? - CanCeylan
2
你不应该关闭grep的WRITE_END和ls的READ_END(与你所做的相反),对吗? - emem
当第一个 execlp() 被更改为 execlp("cat", "cat", "test.c", NULL); 并且下一个被更改为 execlp("tail", "tail", "-n", "3",NULL); 时无法工作。 - Ilaya Raja S
2
示例存在严重缺陷。您没有检查分叉是否失败,并且应始终关闭管道的两端,而不仅仅是一端。否则,试图读取直到EOF的父进程将永远无法接收到EOF。 - user3704639

2

您也可以使用pipe(2,3p)。创建管道,分叉,将管道的适当端口复制到子进程的FD 0或FD 1上,然后执行。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接