@TOC
1. 无名管道
管道是一种半双工的通信方式,数据只能单向流动,而且只能在具有亲缘关系的进程间使用。进程的亲缘关系通常是指父子进程关系。
1.1 特点
- 半双工的(即数据只能在一个方向上流动),具有固定的读端和写端。
- 只能用于具有亲缘关系的进程之间的通信(也是父子进程或者兄弟进程之间)。
- 可以看成是一种特殊的文件,对于它的读写也可以使用普通的read、write 等函数。但是它不是普通的文件,并不属于其他任何文件系统,并且只存在于内存中。
1.2 实质
管道的实质是一个内核缓冲区,进程以先进先出的方式从缓冲区存取数据:管道一端的进程顺序地将进程数据写入缓冲区,另一端的进程则顺序地读取数据,该缓冲区可以看做一个循环队列,读和写的位置都是自动增加的,一个数据只能被读一次,读出以后再缓冲区都不复存在了。当缓冲区读空或者写满时,有一定的规则控制相应的读进程或写进程是否进入等待队列,当空的缓冲区有新数据写入或慢的缓冲区有数据读出时,就唤醒等待队列中的进程继续读写。
2. 实例
// 无名管道
// 适合具有血缘关系的进程进行通信
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/wait.h>
int main()
{
pid_t result;
int r_num;
int pipe_fd[2];
char buf_r[100],buf_w[100];
memset(buf_r,0,sizeof(buf_r));
if(pipe(pipe_fd) < 0)
{
printf("创建管道失败!");
return -1;
}
result = fork();
if(result < 0)
{
perror("创建子进程失败");
exit(0);
}
else if(result == 0)
{
close(pipe_fd[1]);
if((r_num = read(pipe_fd[0],buf_r,100)) >0)
{
printf("从父进程读到的数据为:%s\n",buf_r);
close(pipe_fd[0]);
exit(0);
}
}
else if(result > 0)
{
// 返回的是子进程的pid
close(pipe_fd[0]);
printf("请从键盘输入写入管道的字符串");
scanf("%s",buf_w);
if(write(pipe_fd[1],buf_w,strlen(buf_w))!=-1)
{
printf("父进程向管道写入:%s\n",buf_w);
}
close(pipe_fd[1]);
waitpid(result,NULL,0); //调用waitpid,阻塞父进程
exit(0);
}
}