这里我们先补充一个小知识:
父子进程使用 pipe 进行通信时fork 和 pipe 先后的顺序是先 pipe,后 fork,因为父进程先创建管道,再 fork 创建子进程,这样才能是父子进程共享一块管道的读写操作符。
进行通信的代码如下:
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/wait.h>
int main()
{
int pipefd[2]; // 先创建管道
pipe(pipefd);
int i;
for(i = 0; i < 2; i++) { // 再创建2个子进程
pid_t pid = fork();
if(pid == 0) {
break;
}
}
if(i == 0) // 子进程1 实现 ps aux // 实现写入
{
close(pipefd[0]); // 关闭读端
// ps 会将结果输出至屏幕,即写到了标准输出,因此需要文件描述符重定向,写到管道写端(流入管道)
dup2(pipefd[1], STDOUT_FILENO);
execlp("ps", "ps", "aux", NULL);
perror(“execlp”); // exec调用失败后作错误输出
exit(1);
}
if(i == 1) // 子进程2 gerp // 实现读出
{
close(pipefd[1]); // 关闭写端
dup2(pipefd[0], STDIN_FILENO); // grep是从标准输入读,文件描述符重定向,从管道读
execlp("grep", "grep", "bash", NULL);
perror(“execlp”); // exec调用失败后作错误输出
exit(1);
}
if(i == 2) // 父进程 回收子进程
{
close(pipefd[0]);
close(pipefd[1]); // 关闭两端
pid_t wpid;
while( (wpid = waitpid(-1, NULL, WNOHANG) ) != -1 )
{
if( wpid == 0) continue;
printf(“child died pid = %d\n”, wpid); }
}
return 0;
}
注:若父子间进行通信时,父进程实现 ps,子进程实现 grep时,父进程会使用 execlp 函数,父进程原码会被替换掉,无法回收子进程。