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

文章目录

  • 1、system V共享内存
    • 1、原理
    • 2、模拟实现
      • 关联共享内存
    • 3.共享内存大小
    • 4、共享内存特点


1、system V共享内存

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

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

1、原理

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

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

2、模拟实现

链接: https://gitee.com/kongqizyd/linux-beginner/tree/master/shm

关于获取共享内存的函数,shmget,代码中有写关于参数的用法,不过这个函数创建成功后会返回一个共享内存的标识符,这个下标和文件下标不一样,所以实际上用得少,用文件的方法会更泛用。而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,让系统自己搞。

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读取端会一直读取,所以共享内存没有保护机制(同步互斥)。

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

你可能感兴趣的:(Linux学习,linux,学习,服务器)