进程间通信简介
进程间通信目的:
数据传输:一个进程需要将它的数据发送给另一个进程
资源共享:多个进程之间共享同样的资源。 通知事件:一个进程需要向另一个或一组进程发送消息,通知它(它们)发生了某种事件(如进程终止 时要通知父进程)。
进程控制:有些进程希望完全控制另一个进程的执行(如Debug进程),此时控制进程希望能够拦截另 一个进程的所有陷入和异常,并能够及时知道它的状态改变。
进程间能直接相互传递数据吗?
不可能,进程间是独立的,如果直接传递数据会发生写时拷贝,故需要中间媒介方式来进行通信,让不同进程看到同一份资源。
进程间通信发展:管道 、System V进程间通信 、POSIX进程间通信
进程间通信分类:
管道 :匿名管道pipe ,命名管道
System V IPC : System V 消息队列 、System V 共享内存 、System V 信号量
POSIX IPC: 消息队列 共享内存 信号量 互斥量 条件变量 读写锁
1、管道
管道是Unix中最古老的进程间通信的形式。 我们把从一个进程连接到另一个进程的一个数据流称为一个“管道”
匿名管道:特征是供具有血缘关系的进程,进行进程间通信。(常见于父子)
pipe函数
匿名管道的特性:
1、管道自带同步机制:假设父进程是读端,子进程是写端,如果管道里面没有数据父进程等待子进程写入,如果管道写满了,子进程等待父进程读取数据
2、管道是单向通信的
3、管道是面向字节流的
4、管道只能保证是具有血缘关系的进程间通信,常见于父子
5、管道可以一定程度保证数据读取的原子性,比如写端写入hello world,读端必须等写端写完才能读取。
管道的大小?
可以看到是65536个字节,除以1024等于64,64KB
但是这里是512乘以8字节,是4KB
4kb以内的都是原子性的,大于其是非原子的
命名管道:
管道应用的一个限制就是只能在具有共同祖先(具有亲缘关系)的进程间通信。
如果我们想在不相关的进程之间交换数据,可以使用FIFO文件来做这项工作,它经常被称为命名管道。命名管道是一种特殊类型的文件
进程间通信的本质:
1、要让不同的进程看到同一份资源,匿名管道:父子共享文件的特征,命名管道:文件路径具有唯一性,让进程看到同一个文件
2、通信过程
共享内存
系统V共享内存原理
- 操作系统申请一块内存空间
- OS将该内存映射进对应进程的共享区中(堆栈之间)
- OS可以将映射后的虚拟地址返回给用户
当有大量共享内存时,操作是系统管理其也是通过先描述,在组织的方式,通过大量数据结构来描述共享内存,对共享内存的管理就相当于对数据结构的管理
操作系统管理共享内存的数据结构是
struct{
//key(确定唯一性,两个进程在创建共享内存时传入同一个key时,两个进程就看到了同一份数据结构,通过ftok创建k值)
}
shmat将当前进程和共享内存关联
参数 shmid: 共享内存标识
shmaddr:指定连接的地址
shmflg:它的两个可能取值是SHM_RND和SHM_RDONLY
返回值:成功返回一个指针,指向共享内存第一个节;失败返回-1
shmaddr为NULL,核心自动选择一个地址
shmaddr不为NULL且shmflg无SHM_RND标记,则以shmaddr为连接地址。
shmaddr不为NULL且shmflg设置了SHM_RND标记,则连接的地址会自动向下调整为SHMLBA的整数倍。公式:shmaddr - (shmaddr % SHMLBA)
shmflg=SHM_RDONLY,表示连接操作用来只读共享内存
shmdt函数
shmctl
总结:1、共享内存的生命周期随系统
2、共享内存不提供任何同步与互斥的操作,双方彼此独立
3、共享内存是所有的进程间通信最快的
共享内存的大小最好为4kb的整数倍,否则系统也会给你开大于你申请的内存(4kb整数),但只给你使用你申请的大小。