操作系统为用户所提供的几种进程间通信方式
原因:进程间因为每一个进程都有一个虚拟地址空间,在保证了进程独立性的同时,却使得进程间无法直接通信
因此需要操作系统来提供进程间通信的方式,并且因为通信场景不同,提供的方式也有多种
进程间通信的方式:
1.管道–用于进程间的数据传输
本质:内核当中的一块缓冲区—通过半双工通信实现数据传输(半双工通信–可以选择方向的通信),通过让多个进程都能访问同一块缓冲区,来实现进程间通信
分类:匿名管道、命名管道
1.匿名管道:这块内核中的缓冲区没有标识,因此只能用于具有亲缘关系的进程间通信(父子进程)。
创建管道时,操作系统会提供两个操作句柄(文件描述符),一个用于从管道读数据,一个用于向管道写数据,子进程通过复制父进程的方式,获取到管道的操作句柄进而实现访问同一个管道进行通信
接口:
int pipe(int pipefd[2]);
创建一个匿名管道,向用户通过参数pipefd返回管道的操作句柄
pipefd[0]:用于从管道读取数据
pipefd[1]:用于向管道写入数据
返回值:0–成功,-1 — 失败
特性:若管道中没有数据,则read会阻塞;若管道写满了,则write会阻塞—管道自带同步与互斥
对管道进行数据操作的大小不大于PIPE_BUF == 4096的时候,则保证操作的原子性(原子性就是中间不能被打断)
若管道所有哦写端被关闭,(表示当前没有进程继续写入数据),read读完管道中当前的数据之后就直接返回0
若管道的所有读端被关闭,(表示没有进程读取数据),继续write就会触发异常,程序退出
2.命名管道
内核中的缓冲区,具有标识符(标识符是一个可见于文件系统的管道文件),其他进程可以通过这个标识符,找到这块缓冲区(通过打开同一个管道文件,进而访问到同一块缓冲区),进而实现通信
接口:
int mkfifo(const char* pathname,mode_t mode);
pathname:管道文件名称
mode:文件权限
成功返回0,失败返回-1
打开特性:
若管道文件以只读的方式打开,则会阻塞,知道这个管道文件被以写的方式打开
若管道文件以只写的方式打开,则会阻塞,知道这个管道文件以读的方式打开
若管道文件以读写的方式打开,则不会阻塞
管道的特性:
半双工通信
自带同步与互斥
提供字节流服务—有序,连接,可靠的字节流传输–传输比较灵活