【Linux】进程间通信

进程间通信

进程间因为每个进程都有一个独有的虚拟地址空间,在保证了进程独立性的同时,却使得进程间无法直接通信。为了能使进程间可以通信,操作系统提供了进程间通信的方式,因为通信场景的不同,提供的方法也有多种。

进程间通信的目的

  • 数据传输:一个进程需要将它的数据发送给另一个进程;
  • 资源共享:多个进程之间共享同样的资源;
  • 通知事件:一个进程需要向另一个或者另一组进程发送消息,通知他们发生了某种事件。
  • 进程控制:有些进程需要完全控制另一个进程的执行,此时控制进程希望能够拦截另一个进程的所有陷入和异常,并能够及时知道它的状态改变。

进程间通信方式

进程间通信主要分为两大标准:System V 标准和 POSIX 标准。
关于System V标准的进程间通信主要有以下几类:
1. 管道
管道适用于进程间的数据传输。本质上管道是操作系统在内核中为进程开辟了一块缓冲区,多个进程通过访问同一缓冲区进行通信,数据在缓冲区中以读写的方式被不同进程获取和操作。
管道有三大特性:
1.生命周期随进程;
2.自带同步与互斥;
3.提供字节流服务。
管道主要分为匿名管道和命名管道

  • 匿名管道
    前面说了管道是操作系统为进程在内核中分配的一块缓冲区,匿名管道就是指这块缓冲区没有标识符,因此其他进程无法直接访问匿名管道,只有类似父子进程这样的具有亲缘关系的进程才能使用匿名管道进行通信(原因是子进程会复制父进程的PCB,其中包括这块信息)。匿名管道的创建使用如下接口:
int pipe(int fd[2])

这个接口的作用是创建一个匿名管道,并向用户返回这个管道的操作句柄,其中参数fd[2]中fd[0]用于从管道中读取数据,fd[1]用于从管道中写入数据,如果创建成功返回 0, 创建失败返回 -1.
匿名管道的特性是:
1.若管道中没有数据,使用读操作会阻塞;
2. 若管道写满了,使用写操作会阻塞;
3. 若管道的写端关闭了,读完数据之后不会阻塞而是返回 0;
4.若管道的读端关闭了,继续写入数据会触发异常,程序退出。
这些特性体现了管道自带同步与互斥。

  • 命名管道
    命名管道和匿名管道相反,命名管道是有标识符的一块缓冲区,并且这个标识符一般是一个可见于文件系统的文件。所以命名管道是一个特殊类型的文件,其他进程可以通过这个标识符找到这块缓冲区,即通过打开同一个管道文件,访问到同一缓冲区,进而实现进程间通信。
    创建一个命名管道既可以使用mkfifo filename命令也可以使用接口,函数接口如下:
int mkfifo(const char* pathname, mode_t mode)

参数pathname是命名管道文件的名称,mode是文件权限 。如果创建成功返回0,失败返回-1.
命名管道的打开特性:
1.若管道文件以只读的方式打开,会阻塞,直到这个文件被以写的方式打开;
2.若管道文件以只写的方式打开,会阻塞,直到这个文件被以读的方式打开;
3.若文件以读写的方式打开,就不会阻塞。
不管是匿名还是命名管道,同查那个对管道进行的数据操作的大小不超过PIPE_BUF这个宏的大小,默认是4KB。

2.共享内存
共享内存用于进程间的数据共享,是最快的进程间通信。共享内存的创建大概是以下步骤:首先,在物理内存中开辟一块空间,将这块物理内存映射到程序的虚拟地址空间,进程就可以通过虚拟地址来访问这块内存。多个进程映射到同一物理内存,这样进行通信的方式,不需要进入内核,只需要再共享的内存区进行操作即可。其他方式的通信都是因为内核中的缓冲区,进程在通信的时候会涉及内核态和用户态的两次数据拷贝。而共享内存不会所以速度更快。

  • 共享内存的操作流程:
    1.创建共享内存即开辟具有标识符的物理内存空间;
    2.将共享内存映射到各个进程的虚拟地址空间;
    3.直接通过虚拟地址进行对共享内存的操作;
    4.解除映射;
    5.释放共享内存。
int shmget(key_t key, int size, int flag)  //创建一个共享内存

void* shmat(int shmid, void* addr, int flag) //建立映射

int shmdt(void* shmstart) //解除映射

int shmctl(int shmid, int cmd, struct shmid_ds* bf) //操作共享内存
//cmd参数为IC_RMID的时候是删除共享内存

当删除共享内存的时候,共享内存不会立即被删除(因为可能造成正在访问的进程奔溃)而是将key修改成0,表示这块共享内存不会再接收映射链接,当这块共享内存的映射链接为0的时候,则自动释放。
需要注意的是共享内存自带没有同步与互斥。
3. 消息队列
消息队列用于进程间的数据传输(有标识符)
消息队列实际上就是内核中的一个优先级队列,多个进程通过向同一个队列中 添加 或者 获取 节点来实现通信。主要是传输一个有类型(优先级)的数据块。
特性:
1.自带同步与互斥;
2.生命周期随内核;
3.数据传输自带优先级。

信号量
信号量用于实现进程间的控制,主要是同步和互斥。
本质:内核中的一个计数器(对资源进行计数) + pcb等待队列

互斥的实现:通过只有 0 / 1 的计数器,实现对临界资源访问状态的标记,在访问临界资源
之前先获取信号量,计数 -1;若计数 <0 则使进程等待(将进程pcb加入队列中);否则可
以对临界资源进行访问(并且在访问期间,已经将临界资源的状态置为不可访问状态,因此
可一保证其他进程不会再访问临界资源。当前进程访问完毕之后,则对计数进行+1,则唤醒
一个进程(将一个pcb出队,置为运行状态)

同步的实现:信号量是一个对资源的计数,可以通过计数判断是否能够获取一个资源进行处
理;若计数小于 0,则表示不能获取(并且对计数进行 -1),需要等待(加入pcb队列)。
这时候若其他进程生产一个资源,则会对计数进行 +1,若计数 <= 0 ,则唤醒一个进程。
(负数表示正在等待进程的数量,如果为正说明没有进程需要资源)。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值