位置: 编程技术 - 正文

linux使用共享内存通信的进程同步退出问题(linux的shm共享内存)

编辑:rootadmin

推荐整理分享linux使用共享内存通信的进程同步退出问题(linux的shm共享内存),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:linux系统共享文件命令,linux共享文件,linux共享内存是什么意思,linux如何使用共享文件夹,linux共享内存是什么意思,linux如何使用共享文件夹,linux的共享内存如何实现,linux共享文件,内容如对您有帮助,希望把文章链接给更多的朋友!

两个甚至多个进程使用共享内存(shm)通信,总遇到同步问题。这里的“同步问题”不是说进程读写同步问题,这个用信号量就好了。这里的同步问题说的是同步退出问题,到底谁先退出,怎么知道对方退出了。举个例子:进程负责读写数据库A,进程B负责处理数据。那么进程A得比进程B晚退出才行,因为要保存进程B处理完的数据。可是A不知道B什么时候退出啊。A、B是无关联的进程,也不知道对方的pid。它们唯一的关联就是读写同一块共享内存。正常情况下,进程B在共享内存中写个标识:进程A你可以退出了,也是可以的。不过进程B可能是异常退出,连标识都来不及写。其次,共享内存用来做数据通信的,加这么个标识感觉不太好,有滥用的感觉。

  采用socket通信没有这个问题,因为进程B退出怎么也会导致socket断开,哪怕是超时。但shm却没有协议来检测这些行为,如果自己也做一个未免太麻烦。那就从共享内存下手吧。

  共享内存是由内核来管理的,一个进程删除本身打开的共享内存并不影响另一个进程的共享内存,哪怕都是同一块共享内存。这是因为共享内存在内核中一个引用计数,一个进程使用该共享内存就会导致引用计数加1。如果其中一个进程调用了删除函数,只有这个计数为0才会真正删除共享内存。那么,需要最后才退出的进程检测这个计数就可以了。

  在System V的共享内存中,创建一个共享内存会初始化一个结构:

复制代码代码如下:struct shmid_ds { struct ipc_perm shm_perm; /* Ownership and permissions */ size_t shm_segsz; /* Size of segment (bytes) */ time_t shm_atime; /* Last attach time */ time_t shm_dtime; /* Last detach time */ time_t shm_ctime; /* Last change time */ pid_t shm_cpid; /* PID of creator */ pid_t shm_lpid; /* PID of last shmat(2)/shmdt(2) */ shmatt_t shm_nattch; /* No. of current attaches */ ... };

使用shmctl函数可以读取该结构体,其中的shm_nattch就是使用该共享内存的进程数。

  不过,现在有了新的POSIX标准,当然要用新标准了。shm_open创建的共享内存也具有“一个进程删除本身打开的共享内存并不影响另一个进程的共享内存”的特点。可是用shm_open创建的共享内存不再有上面的结构,那么,内核是怎么管理shm_open创建共享内存??看下面的源码:

linux使用共享内存通信的进程同步退出问题(linux的shm共享内存)

复制代码代码如下:/* shm_open - open a shared memory file */</p><p>/* Copyright , Red Hat Inc. */</p><p>#include <sys/types.h>#include <sys/mman.h>#include <unistd.h>#include <string.h>#include <fcntl.h>#include <limits.h></p><p>intshm_open (const char *name, int oflag, mode_t mode){ int fd; char shm_name[PATH_MAX+] = "/dev/shm/";</p><p> /* skip opening slash */ if (*name == '/') ++name;</p><p> /* create special shared memory file name and leave enough space to cause a path/name error if name is too long */ strlcpy (shm_name + 9, name, PATH_MAX + );</p><p> fd = open (shm_name, oflag, mode);</p><p> if (fd != -1) { /* once open we must add FD_CLOEXEC flag to file descriptor */ int flags = fcntl (fd, F_GETFD, 0);</p><p> if (flags >= 0) { flags |= FD_CLOEXEC; flags = fcntl (fd, F_SETFD, flags); }</p><p> /* on failure, just close file and give up */ if (flags == -1) { close (fd); fd = -1; } }</p><p> return fd;}

我嚓,这就是创建一个普通的文件啊,只是创建的位置在/dev/shm下(也就是RAM上)。再来看看删除共享内存的函数shm_unlink:

复制代码代码如下:/* shm_unlink - remove a shared memory file */</p><p>/* Copyright , Red Hat Inc. */</p><p>#include <sys/types.h>#include <sys/mman.h>#include <unistd.h>#include <string.h>#include <limits.h></p><p>intshm_unlink (const char *name){ int rc; char shm_name[PATH_MAX+] = "/dev/shm/";</p><p> /* skip opening slash */ if (*name == '/') ++name;</p><p> /* create special shared memory file name and leave enough space to cause a path/name error if name is too long */ strlcpy (shm_name + 9, name, PATH_MAX + );</p><p> rc = unlink (shm_name);</p><p> return rc;}

这也只是一个普通的unlink函数。也就是说,POSIX标准的共享内存就是一个文件。所谓的“一个进程删除本身打开的共享内存并不影响另一个进程的共享内存”就相当于你用fstream对象打开了一个文件,然后去文件夹把文件删除了(也就是对文件进行了unlink操作),可是fstream对象还可以正常读写文件,并没有什么引用计数。这下好了,进程退出时又没法同步了。

  不过,在linux下怎么会有解决不了的问题呢?解决不了只能说明自己太菜。既然是文件,那就从文件下手。那文件有什么是原子操作,又可以计数的呢。答案:硬链接。比如:

复制代码代码如下:xzc@xzc-HP-ProBook-s:/dev/shm$ stat abc 文件:"abc" 大小:4 块:8 IO 块: 普通文件设备:h/d Inode: 硬链接:1权限:(/-rw-rw-r--) Uid:( / xzc) Gid:( / xzc)最近访问:-- ::. +最近更改:-- ::. +最近改动:-- ::. +创建时间:-xzc@xzc-HP-ProBook-s:/dev/shm$

这个硬链接可以通过fstat函数获取。可是要这样实现的话,意味着需要先创建一块共享内存,每个进程引用的时候需要调用link函数来创建一个硬链接。问题解决了,可是这样会在/dev/shm下多个N多个文件。这可是RAM啊,虽然现在的服务器都比较牛,但这样做也不太好吧。好吧,还有一个flock文件锁。flock使用LOCK_SH参数多个进程对同一个文件加锁。这样,进程B初始化共享内存时加锁(可以有多个这样的进程),在退出(包括异常退出)时解锁。进程A在退出时检测这个锁。当发现无锁时说明可以安全退出了。

  同步退出的问题基本解决了。来不及写代码去验证,下次吧。

PS:内核unlink时应该也是有计数才知道当前有没有进程打开文件,在什么时候应该删除文件。这个还得去查资料,看用不用得上。另外lsof这个工具是可以检测到所有打开该共享内存的进程及相应的状态。这个应该也是有对应的api的,只是现在还没搞懂。

Linux系统下如何配置安装SSH服务?如何开启SSH服务? SSH作为Linux远程连接重要的方式,如何配置安装linux系统的SSH服务,如何开启SSH?下面来看看吧(本例为centos系统演示如何开启SSH服务)工具/原料linuxcent

Linux下swap耗尽该怎么办?如何释放swap? Linux下交换内存耗尽如何查看及如何处理。以及查看shell脚本描述。交换内存不足时,启动应用时会报内存不足。1、启动应用时报内存不足,直接Linux命

Linux chmod命令及权限含义 对于一条权限赋值命令:sudochmod-R;我们必须了解其含义:(1)各字段含义:sudochmod-R(更改文件夹及其子文件夹)7(所有者权限)6(组用户权限)4

标签: linux的shm共享内存

本文链接地址:https://www.jiuchutong.com/biancheng/368224.html 转载请保留说明!

上一篇:linux下通过ntp时间同步方法(linux ntp查看)

下一篇:Linux系统下如何配置安装SSH服务?如何开启SSH服务?(linux系统怎样)

  • 小规模纳税人发票可以抵扣吗
  • 库存股增加为什么资产减少
  • 车辆计提折旧需要计提吗
  • 印花税退税流程怎么操作
  • 公司给非员工代缴社保合法吗
  • 企业购入生产设备所支付的增值税
  • 营改增后预存话费可以报销吗?
  • 定金不买了可以退吗
  • 停车场需要对车辆负责吗
  • 租赁合同印花税双方都要交吗
  • 普通发票和增值税发票都能报销吗
  • 营改增后建筑业开票规定
  • 上月作废的发票忘了作废,怎么做账
  • 资产置换会计处理办法
  • 补缴以前税收滞纳金多少
  • 有限责任公司向合伙企业投资
  • 向其他股东购买股权
  • 分支机构怎么认定和纳税?
  • 支付股息如何做账务处理
  • 交易性金融资产公允价值变动计入
  • 去年的发票今年能用吗
  • win10如何禁用administrator用户
  • 汇算清缴工资薪金支出包含社保费吗
  • win11预览版更新界面
  • php实现的三个常数是什么
  • 盈余公积的提取基数
  • 前端vue实现预览excel
  • 上半年的业务总结几句话
  • 上个月做无票收入这个月开票,怎么做分录
  • 以太网默认网关怎么查看
  • Win11 Build 22000.348更新补丁KB5007262预览版发布(附更新修复内容汇总)
  • php 统计
  • 看望员工家属
  • framework4.0怎么打开
  • idea如何运行vue项目
  • elementui表格自定义排序
  • unrar解压命令
  • 小企业原材料
  • 退役士兵抵税2019
  • 还没报税就自动清卡了怎么办
  • 企业办理土地证需要哪些资料和手续
  • mysql使用中遇到的困难和问题
  • 什么是服务型
  • 收回已冲销的应收账款会计分录
  • 工地会计内账
  • sqlserver2005没有实例
  • mysql获取字段值的长度
  • 信息技术服务增值税税负率是多少
  • 贸易中的发票抵押和发票融资
  • 自产货物赠送客户账务处理
  • 个人所得税经营所得税申报表A表
  • 营业外收入核算的内容有
  • 职工薪酬的范围及内容
  • 工会经费交纳
  • 会计一般采用什么科目
  • 一个简单的微笑说说
  • 在sql server中触发器不具有什么类型
  • mysql优化十大技巧
  • linux中fi
  • freebsd操作命令
  • 使用linux常用命令
  • 怎么关闭windows登录密码
  • macbook graphpad
  • windows xp输入法不见了怎么办
  • 影响电脑速度的因素
  • 硬盘分区win7怎么扩充c盘
  • windows10周年纪念版
  • linux 运行二进制文件
  • 查看电脑是否可以装两个固态
  • 解决Extjs 4 Panel作为Window组件的子组件时出现双重边框问题
  • 用简洁的语言推荐一本书
  • 批处理编程教程
  • relative absolute无法冲破的等级问题解决第1/3页
  • 怎么通过微信号查到对方的手机号码
  • 批处理结束后不关闭窗口
  • python基础教程电子版
  • js教程
  • python 自定义类模块
  • jQuery中通过ajax调用webservice传递数组参数的问题实例详解
  • 湖南省国家税务局历任局长
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设