技术标签: Linux 进程间通信 linux 共享内存 IPC
本文介绍另一种进程间通信方式:共享内存。
什么是共享内存?
共享内存就是申请了一块物理内存用来进行数据共享,需要进行数据共享的进程可以将同一块物理内存映射到自己的虚拟地址空间,然后通过自己的虚拟地址空间直接访问这块空间,从而实现数据交流。
共享内存和管道的区别:
(1)管道的通信类型是传输,数据是先进先出的,数据被读取走之后才会消失,因此数据不会被覆盖。
(2)共享内存是覆盖式的,因此如果多个进程对共享内存进行操作的话,就会存在安全隐患。但是共享内存也是最快的进程间通信方式。
共享内存与管道在数据传输效率中的差异:
(1)管道:数据在传输过程中经历了两次数据在内核空间与用户空间之间的拷贝。
(2)共享内存:进程直接通过虚拟地址空间访问共享内存,不需要进行内核空间与用户空间的两次数据拷贝。
如图:进程A和进程B之间如果使用管道进行数据传输,那么首先进程A要把数据从用户空间拷贝一份发送到内核空间的管道中。然后进程B要从内核空间中拷贝一份数据到用户空间中,这样才可以使用。因此管道传输一次数据要经历两次在内核空间与用户空间之间的拷贝。
如图:共享内存是在物理内存中申请的一块空间,需要共享数据的两个进程都与这块共享内存建立映射关系,然后就可以直接从共享内存中读取数据,而不需要经历两次拷贝,因此是最快的进程间通信方式。
创建或打开共享内存:
将共享内存映射到虚拟地址空间:
共享内存操作:
解除映射关系
删除共享内存
在删除共享内存的时候,还有一个东西要注意:映射连接数(和这块共享内存建立映射的进程的个数)。
进程解除和共享内存的映射关系后,映射连接数减1。但是一个共享内存通常是多个进程一起操作的(比如进程A,进程B),进程A执行了删除共享内存的操作后,这个共享内存是不会被立即删除的,不然进程B用的好好的突然断了咋整。
因此执行了删除共享内存这操作后,共享内存不能被立即删除,但是共享内存会被标记成被删除的状态,以后其他进程如果要连接这个共享内存,系统发现该共享内存是被删除状态,就拒绝其他进程连接了。
等到连接这块共享内存的进程都和这块共享内存解除映射后,映射连接数就变成了0,此时系统才会释放共享内存。
按照共享内存的操作流程来依次介绍共享内存的操作接口。
int shmget(key_t key,size_t size,int shmflag)
作用:创建或打开共享内存
ket_t key:共享内存的标识符,也就是共享内存的名字(注意把标识符和文件标识符区分开)。
size_t size:要创建的共享内存的大小。
int shmflag:创建权限/打开方式。如果是创建共享内存就是创建权限,如果是打开共享内存就是打开方式。打开方式通常是 IPC_CREAT(共享内存不存在就创建)
返回值:成功返回共享内存的操作句柄(非负整数),失败返回-1
关于key:
多个进程通过共享内存的名字找到同一个共享内存。(如果把名字命名为 IPC_PRIVATE,就代表这个共享内存只能用于具有亲缘关系的进程间通信,因为这样其他进程就找不到这个共享内存)
关于size:
共享内存的大小和共享内存的占用空间不是一回事。共享内存空间的开辟是以内存页为单位的,不是说要创建10字节的共享内存就会开辟10字节大小的共享内存空间。因为内存和磁盘的空间管理是以块为单位管理的,而不是以字节为单位管理。
也就是说,如果要创建10字节的大小的共享内存,实际上会开辟4096个字节大小的共享内存,只不过使用的大小是10字节。
void* shmat(int shmid,void* addr,int shmflag)
作用:建立映射关系
返回值:建立成功则返回映射的空间的首地址(通过这个地址操作共享内存),失败返回(void*)-1。
因为共享内存其实就是一块内存,因此我们可以直接用操作内存的函数来操作共享内存,比如:memcpy、strcpy、printf
int shmdt(void* shm_start)
作用:解除进程与指定的共享内存的映射关系
int shmctl(int shmid,int cmd,struct shmid_ds* buf)
作用:删除共享内存(这个函数不止是用来删除,它的功能很多,只不过我们常用到的是删除功能)
返回值:返回值其实不是固定的,因为不同的操作选项代表了不同的功能,不同的功能有不同的返回值。对于删除操作来说,成功返回0,失败返回-1。
如图:此时系统中只有一个共享内存。
注意,此时的共享内存状态改变为 dest(被删除状态),变成这个状态后就不允许其他进程再和它建立映射了。
文章浏览阅读383次。根据邱维声老师的高等代数课程,整理的笔记。_比较消元法和初等变换求线性方程组的异同,并阐述自己的收获
文章浏览阅读281次。Android HyBridge 开发一、三种App开发方式对比1. Native App特点:UI元素、数据内容、逻辑架构都安装在手机终端,导致不可跨平台,每次版本升级都要重新打包。缺点:无法跨平台、升级麻烦、开发成本高(指跨平台开发成本高)优点:速度快,用户体验好。2. Web App定义:可理解为移动端的网站,将网页部署在服务器上,用户通过各大浏览器来访问。缺点:页面访问速度慢、用户体验差。..._安卓应用商店app demo
文章浏览阅读318次。Stable Diffusion 图生图知识思维导图;使用 5W1H 框架启动一个可控的AI项目;培生集团将生成式AI学习工具引入在线高等教育平台;前 Meta AI 高管离职创业,做教育类 ChatGPT 应用……点击阅读全文_inscode deecamp x csdn ai应用创新大赛
文章浏览阅读814次,点赞9次,收藏7次。作者简介:我目前是一个在校学生,现在不敢说自己擅长什么,但是我想通过自己的学习努力让自己的技术、知识都慢慢提升,希望我们一起学习呀~。有话想说:写博客、记笔记并不是一种自我感动,把学到的东西记在脑子里才是最重要的,在这个过程中,不要浮躁,希望我们都可以越来越优秀!由于算法不会改变原有的元素集合,只需要一个额外的变量控制索引变化,所以空间复杂度为常数级:O(1)️兴趣领域:目前偏向于前端学习 算法学习。语言说明:代码实现我会用Python/C++~空间复杂度:O(1)
文章浏览阅读167次。【2016.07】认知无线电天线设计Antenna design for cognitive radio,共320页。如果需要电子版,请联系QQ:3042075372。本书从天线设计的角度阐述了认知无线电,并以一种协议的形式引入了认知无线电的概念,该协议受益于频谱中未被充分利用的区域。This one-of-a-kind new resource presents cognitive ra..._底层认知无线电
文章浏览阅读9.7k次,点赞5次,收藏29次。date命令来自于英文单词它自己,也就是时间、时钟的意思,其功能是用于显示或者设置系统日期与时间信息的。运维人员可以根据自己需要的格式来输出系统时间信息。_date设置当前日期
文章浏览阅读388次。pNewRaster是你的Raster图层IRasterBandCollection pRasterBC =(IRasterBandCollection ) pNewRaster;IRasterBand pRasterBand = pRasterBC.Item(0);ITable pTable = pRasterBand.AttributeTable;IQueryFilter pQueryFilter=new QueryFilterClass ();pQueryFilter .WhereClau._r raster getvalue
文章浏览阅读226次。如果你曾经写过或者用过 Python,你可能已经习惯了看到 Python 源代码文件;它们的名称以.Py 结尾。你可能还见过另一种类型的文件是 .pyc 结尾的,它们就是 Python “字节码”文件。(在 Python3 的时候这个 .pyc 后缀的文件不太好找了,它在一个名为__pycache__的子目录下面。).pyc文件可以防止Python每次运行时都重新解析源代码,该文件大大节省了时间。..._python反编译class文件
文章浏览阅读38次。本题是一个经典的动态规划题。直接利用记忆化搜索:见图解Ac code :#include<stdio.h>#include<string.h>#define max(i,j) (i>j?i:j)#define maxn 105int a[maxn][maxn];int d[maxn][maxn];int s(int i,...
文章浏览阅读2.2k次。VMware Workstation 10.0.7 安装VMware Workstation 10.0.7 官网下载地址1、点击“exe.”开始安装,选择“下一步”;2、选择“接受”,点击“下一步”;3、选择“典型”,可更改安装路径;4、取消“启动时检查产品更新”、“帮助”选项;5、默认选项,点击“下一步”;6、正在安装VMware;7、输入密钥;8、安装成..._vmware workstation 10.0.7
文章浏览阅读380次。一旦定义了最终结果DataFrame / Dataset,剩下的就是开始流式计算。为此,必须使用返回的 DataStreamWriter Dataset.writeStream()。_pyspark structured streaming 打印
文章浏览阅读863次,点赞28次,收藏12次。/ 链式结构:表示队列}QNode;// 队列的结构//头指针//尾指针int size;//数据个数}Queue;我们先定义节点用来存储指向下一个节点的指针和要保存的数据,然后定义指向队列的头指针和尾指针和存储数据的个数。// 初始化队列// 队尾 入队列// 队头 出队列// 获取队列头部元素// 获取队列队尾元素// 获取队列中有效元素个数// 检测队列是否为空,如果为空返回非零结果,如果非空返回0// 销毁队列。