管道
管道是一种最基本的IPC机制,作用于有血缘关系的进程之间,完成数据传递。调用pipe系统函数即可创建一个管道。有如下特质:
1.其本质是一个伪文件(实为内核缓冲区)
2.由两个文件描述符引用,一个表示读端,一个表示写端。
3.规定数据从管道的写端流入管道,从读端流出。
管道的原理: 管道实为内核使用环形队列机制,借助内核缓冲区(4k)实现。
管道的局限性:
① 数据自己读不能自己写。
② 数据一旦被读走,便不在管道中存在,不可反复读取。
③ 由于管道采用半双工通信方式。因此,数据只能在一个方向上流动。
④ 只能在有公共祖先的进程间使用管道。
常见的通信方式有,单工通信、半双工通信、全双工通信。
管道的特点:
(1)管道必须读,写进程同时打开,否则会阻塞;
(2)如果管道没有数据,read会阻塞;
(3)管道的写端关闭,读read返回值为0(下节课演示)
(4)管道打开只有只读和只写两种方式;读写方式打开是未定义的;
管道的实现
循环队列
如果管道是空的,读操作会阻塞;如果管道是满的,写操作会阻塞;
有名管道:有名管道
无名管道
无名管道只能在父子进程间通讯;
无名管道,pipe创建无名管道;它只能进行父子间的通信;
int pipe(int fds[2]);
//调用pipe()创建一个管道,返回两个文件描述符,fds[0]为读端,fds[1]为写端
无名管道的通信:
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<assert.h>
int main()
{
int fd[2];
assert(pipe(fd)!=-1);
pid_t pid=fork();
assert(pid!=-1);
if(pid==0)
{
close(fd[1]);
char buff[128]={0};
read(fd[0],buff,127);
printf("chlid read:%s",buff);
close(fd[0]);
}
else
{
close(fd[0]);
write(fd[1],"hello",5);
close(fd[1]);
}
exit(0);
}