Linux进程之间的通信--管道
进程之间的通信
什么是IPC?
进程间通信,InterProcess Communication
进程间通信常用的4中方式
管道----------简单
信号----------系统开销小
共享映射区----(有无学院关系的进程间通信都可以)
本地套接字-----稳定
什么是管道?
本质:
内核缓冲区
伪文件-不占用磁盘空间
特点:
两部分:
读端、写端,对应两个文件描述符
数据写端流入,读端流出
操作管道的进程被销毁之后,管道被自动释放
对管道的操作,默认阻塞。
管道的原理
内部实现方式:队列
环形队列
特点:先进先出
缓冲区大小:
默认4k
但也会根据实际情况适当调整
管道的局限性
队列:
数据只能读取一次,不能重复读,读完之后就销毁了。
半双工:
单工:只能一方向另一方发送
半双工:双方都能发送和接受,但是同一时刻只能一方发送
双工:同时双发送
匿名管道:
适用于有血缘关系的进程。
那么什么样的有血缘关系呢?
父子、爷孙、叔侄,因为fork是对父进程的赋值,除了pcbid不一样其它都已一样。包括文件描述符表
如何创建匿名管道?
int pip(int fd[2]);
//fd-传出参数
//fd[0] //读端
//fd[1] //写端
程序示例
#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<string.h>
int main(int argc,const char * argv[]){
int fd[2];
int ret=pipe(fd);
if(ret==-1){
perror("pipe error");
exit(1);
}
printf("pipe读端[0]=%d\n",fd[0]);
printf("pipe写端[1]=%d\n",fd[1]);
close(fd[0]);
close(fd[1]);
return 0;
}
父子进程使用管道通信
单个进程是可以使用管道完成读写操作的
如果父进程写的慢,子进程读的快,那么子进程需不需要停下来等待父进程写?
不需要,因为管道默认是阻塞的。子进程将管道中的数据读完只能等父进程写完在读。
有这么一种说法:父进程读,要关闭写端;子进程写,要关闭读端
。其实这只是一种习惯,防止误操作,不是必须的,还有就是如果不关闭,可能会浪费系统资源。
history | grep python
中间的"|"就代表一个匿名管道
这个意思事history得到的数据输入到匿名管道中
然后grep从管道中查找数据
那么如果用程序实现呢?
分析一下场景,history父进程调用执行,grep python 子进程调用执行。他们用同一个管道进行通信。
如何用程序申请的管道呢?
想一下如果单独执行history它的输出端在标准输出文件描述符STDOUT_FILENO让后输出到终端
grep python从标准输入文件描述符STDIN_FILENO搜索数据。
下面画图分析一下。
#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<string.h>
int main(){
int fd[2];
int ret=pipe(fd);
if(ret==-1){
perror("pip error");
exit(1);
}
pid_t pid=fork();
if(pid==-1){
perror("fork error");
exit(1);
}
//父进程
printf("pipe[0]=%d\n",fd[0]);
printf("pipe[1]=%d\n",fd[1]);
//父进程
if(pid>0){
//写管道的操作,关闭读端口
close(fd[0]);
//文件描述符的重定向
// stdout_fileno--->管道的写端
dup2(fd[1],1);
//dup2(fd[1],STDOUT_FILENO);
//执行ls;
execlp("ls","ls",NULL);
perror("ls execlp");
exit(1);
}
else if(pid==0){
close(fd[1]);
dup2(fd[0],0);
//dup2(fd[0],STDIN_FILENO);
execlp("grep","grep","python","--color=auto",NULL);
perror("grep execlp");
exit(1);
}
printf("pipe[0]");
close(fd[0]);
close(fd[1]);
return 0;
}
执行效果一样
管道的读写行为
读操作
有数据
read(fd,)-----正常读,返回读出的字节数
无数据
写端全部关闭
read 解除阻塞,返回0
相当于读文件读到了尾部
没有全部关闭
read阻塞
写操作
读端全部关闭
管道破裂,进程被终止
内核给当前进程发信号,SIGPIPE,终止进程
读端口没有全部关闭
缓冲区写满了
write阻塞
缓冲区没有满
write继续写
如何设置非阻塞?
默认读写端都阻塞
设置读端为非阻塞pipe(fd[])
fcntl---变参函数
复制文件描述符--dup
修改文件属性---open的时候对应flag属性
设置方法:
获取原来的flags
int flags=fcntl(fd[0],F_GETFL);
//设置新的flags
flag |= O_NONBLOCK;//flag=flag | O_NONBLOCK
fcntl(fd[0],F_SETFL,flags);
查看缓冲区大小:
有名管道 fifo(first in first out)
1、特点
有名管道
在磁盘上有这个一个文件ls -l
伪文件,在磁盘上的大小为0
在内核中有一个对应的缓冲区
半双工的工作方式
有名管道也是阻塞的
2、使用场景
没有血缘关系的进程通信,使用fifo
3、创建方式
mkfifo name
4、fifo文件可以使用IO函数操作
open/close
read/write
5、进程间通信
fifo文件 —myfifo
两个不相干的进程 A(a.c) B(b.c)
a.c------>read
int fd=open("myfifo",O_RDONLY);
read(fd,buf,sizeof(buf));
close(fd);
b.c—>write
int fd1=open("myfifo",O_WRONLY);
write(fd1,"hello world",11);
close(fd1);