Linux学习记录——이십 进程间通信(2)共享内存


1、system V共享内存

system是一套标准,独立于文件系统之外,是系统专门为通信而设计出来的内核模块,称之为system V的IPC通信机制。

共享内存的主要做法也是让两个毫不相关的进程看到同一份资源。

1、原理

进程的地址空间内,栈和堆区之间有个共享区,堆是向上增长,栈是向下增长,重合的那个地方就是共享区,也就是动态库存放的位置。在物理内存开辟一块空间,将这个空间通过映射到共享区里,然后把这块空间的起始地址返回给用户;同样地,另一个进程也这么做,这样两个进程就可以访问到同一个物理内存内的空间了。

如果不再继续共享了,那么进程可以取消掉映射关系,也就是修改页表,然后把地址空间的共享内存给释放掉,最后把物理内存的共享内存释放,就不再共享了。

2、模拟实现

整体代码

对照着下面的话来分析代码,从server.cc文件开始看。

在这里插入图片描述

shmget,申请一个System V的共享内存,是个系统调用接口,size就是申请的内存块的大大小,shmflg是大写字母,用来实际操作,代码中有写。这个函数创建成功后会返回一个共享内存的标识符,这个下标和文件下标不一样,所以实际上用得少,用文件的方法会更泛用。而key参数需要用ftok函数来设定,ftok结合传过来的项目id和路径来生成key。

上面写到了共享内存的原理,在系统中不止有一个共享内存,也不止有两个进程在共享内存,所以系统中会同时存在很多个内存,系统对他们的管理就是先描述再组织,创建一个结构体来存储共享内存的所有属性。

对于底层来说,对于共享内存的管理就变成了对结构体的管理,只要两个进程可以得到一样的数据,那就访问到了一样的共享内存。但是系统是存在很多个共享内存的,两个进程如何确保访问的是一个内存?这里就是用到了ftok函数的key,一个进程创建共享内存,指定了路径,传入了id,就会形成一个key,key会填入共享内存的结构体里,而另一个进程也调用ftok函数,它如果用同样的路径和id,也会生成一个key,然后这两个key去匹配,这是一定相等的,所以就会访问到同一个资源。

共享内存相关的进程退出了,但内存还在。要如何确定共享内存存在?

ipcs

看Shared Memory Segments那栏,也可以后面加上-m只看这栏。ipcrm 可以用来删共享内存,但应当按照key还是shmid来删?

shmid是shmget函数返回的共享内存的标识符,相当于fd。key是在系统中用的,相当于共享内存的inode。在用户层都用shmid。指令就在用户层,系统接口层之上。删除指令ipcrm -m shmid。

shmctl

这也反映了一个问题,共享内存是不随进程的,把写的模拟实现的文件生成可执行程序,调用又清除,再次调用,server会失败,因为不需要再创建了,而client可以获取和上次一样的。共享内存是随操作系统的。除了指令删除,还可以调用系统接口。

虽然创建了共享内存,但不能直接使用,要用shmat函数。

在这里插入图片描述

用户得到的共享内存的地址是虚拟地址。shmat参数里的shmaddr是指放在地址空间的哪里,所以一般设为None,让系统自己搞。shmflg可以设置内存块的属性,为0就是可读可写。

3.共享内存大小

上面的代码中有设置gsize = 4096,如果一边运行,一边用ipcs -m来查看进程,可以看到我们开的共享内存大小是4096。这个大小可以改,共享内存的大小是以PAGE页为单位的,它就向上对齐到PAGE,系统分配内存是以页为单位的,而这个页是4kb的大小,也就是文件和磁盘数据块进行IO时的单位。但是如果我们设定gsize为4097时,共享内存的大小应当是4096 * 2,因为4097超过了4096,按照对齐到PAGE的规则,应该是2倍的4096,但现在用命令查看却是4097。事实上,确实扩容了一下,但用户用多少系统给多少,系统确实对齐到2倍的4096了,但我只用4097,共享内存大小就是4097。

------------------------------------------------------------------------------------------------------------------------------------------------

4、共享内存特点

共享内存没有用到系统调用接口,这是因为共享内存映射到了进程地址空间,它已经被看到了,不需要用户做什么。正是因为有这种特点,可以让进程通信的时候,减少拷贝次数,所以共享内存是所有进程间通信最快的

按照上面的代码,如果client没有调用,共享内存里没有内容,只调用了server,server读取端会一直读取,会一直打印空,而管道会让读端阻塞,等到写端开始写才能继续读,所以共享内存没有保护机制(同步互斥)。

为什么共享内存有这个看似挺致命的缺点?管道是通过系统接口进行通信的,系统接口在系统内部被调用时会管理数据;共享内存直接通信。

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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值