IPC共享内存
shmget
函数创建一片共享内存空间,返回这片共享内存的id。
shmat
调用能将这片共享内存空间映射到自己进程的虚拟内存空间,返回虚拟地址。
两个进程要通过共享内存进行通信,必然要映射到同一片物理内存。如何保证这一点呢?shmget
函数有一个参数是shm_key
,它能唯一的标志一片物理内存。第一个进程用某个key创建了共享内存空间,而另一个进程使用相同的key调用shmget
时,不会创建新的共享内存,而是直接返回前一个进程创建的。
IPC共享内存实际上是使用了内核的tmpfs文件系统。这个文件系统是在物理内存上的。
POSIX共享内存
shm_open
返回一个共享内存对象。而这个共享内存对象实际上就是tempfs文件系统中的一个文件。
直接使用mmap
将这个文件映射到自己的虚拟内存空间。
同样shm_open
中有一个参数是name
,它可以唯一地标志一个tempfs文件。于是只要两个进程将同一个tempfs文件映射到自己的虚拟内存空间,就能实现通信。
两种共享内存的区别
posix将tempfs文件加入到了进程的打开文件表中,一个进程要关闭共享内存对象,使用的是unlink
函数,目的是减少文件的引用计数,在文件的引用计数减小为0的时候,这个文件就会被关闭。或者当前进程退出,这个文件引用计数也会减1。
而IPC的共享内存虽然实际上也是用tempfs文件系统,但是没有将文件关联到进程,进程的打开文件表中没有这个文件。当前进程退出时,这个文件的引用计数不会减1。除非显式地销毁共享内存,或者电脑关机,共享内存会一直存在。