文章出处:http://blog.csdn.net/yake25/article/details/7447086
I/O重定向和管道:使用pipe、fork、exec详解。
要使 who | sort 用程序实现,需要进行 I/O重定向,参考了Unix / Linux编程实践教程,画了几个图来理解这个程序,可能会更好。
1. 当运行pipe( pipe_id )以及fork之后,父进程和子进程共享stdin、stdout和管道,如图所示:
2. 我们知道,可以设定子进程写,父进程读,也就是子进程执行who程序,通过写管道,输出到父进程的读管道口(这是没有重定向的情况),于是万恶的重定向来了,他告诉你可以进入管道,但是不是常规的那个门口进入,而是从另外一个门进入。但管道还是管道,只是进出口不同而已,你懂得。
第一步:关闭子进程的读管道口
第二步: 关闭子进程输出流
第三步: 复制子进程的写管道
第四步: 关闭子进程的写管道
至此,成功实现子进程写管道口角色的转换,于是在子进程执行程序,执行比如who时,他以为他输出到标准输出流,其实他已经输出到了写管道口。
3. 同理,对父进程进行一系列类似的处理
至此,成功实现角色转换;
代码:
#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);
}