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