O重定向和管道:使用pipe、fork、exec详解

要使who|sort用程序实现,需进行I/O重定向,参考了Unix/Linux编程实践教程,画了几个图来理解这个程序,可能会更好。

  1. 当运行pipe(pipe_id)以及fork之后,父进程和子进程共享stdin、stdout和管道,如图所示:
多图详解I/O重定向和管道:使用pipe、fork、exec详解

  1. 我们知道,可以设定子进程写,父进程读,也就是子进程执行who程序,通过写管道口,输出到父进程的读管道口(这是没有重定向的情况),于是万恶的重定向来了,他告诉你可以进入管道,但是不是常规的那个门口进入,而是从另外一个门进入。但是管道还是管道,只是进出口不同而已,你懂得。

第一步:关闭子进程的读管道口

多图详解I/O重定向和管道:使用pipe、fork、exec详解

第二步:关闭子进程输出流

多图详解I/O重定向和管道:使用pipe、fork、exec详解

第三步:复制子进程写管道口

多图详解I/O重定向和管道:使用pipe、fork、exec详解

第四步:关闭子进程写管道

多图详解I/O重定向和管道:使用pipe、fork、exec详解

至此,成功实现子进程写管道口角色的转换,于是在子进程执行程序,执行比如who时,他以为他输出到标准输出流,其实他已经输出到了些管道口。


  1. 同理,对父进程进行一系列类似的处理
多图详解I/O重定向和管道:使用pipe、fork、exec详解

多图详解I/O重定向和管道:使用pipe、fork、exec详解

多图详解I/O重定向和管道:使用pipe、fork、exec详解

多图详解I/O重定向和管道:使用pipe、fork、exec详解

至此成功实现角色转换!


代码

<code>#include "../apue.h" int main(int argc, char *argv[]){int pid, pipe_id[2], newfd;if(argc != 3)oops("usage: cmd need 3 paraments", 1);if(pipe(pipe_id) < 0)oops("pipe", 1);if((pid = fork()) < 0){oops("fork", 1);}else if(pid == 0)///子进程负责写{close(pipe_id[0]);if(dup2(pipe_id[1], 1) == -1)oops("dup2", 1);close(pipe_id[1]);execlp(argv[1], argv[1], NULL);oops("execlp", 2);}else///父进程负责读{close(pipe_id[1]);if(dup2(pipe_id[0], 0) == -1)oops("dup2", 1);close(pipe_id[0]);execlp(argv[2], argv[2], NULL);oops("execlp", 1);}exit(0);}/<code>

https://blog.csdn.net/yake25/article/details/7447086


分享到:


相關文章: