文章目录
一、共享内存的定义和原理
1、共享内存的定义
顾名思义,共享内存就是允许多个不相关的进程访问同一个逻辑内存。共享内存是在两个正在运行的进程之间共享和传递数据的一种非常有效的方式。不同进程之间共享的内存通常安排为同一段物理内存。进程可以将同一段共享内存连接到它们自己的地址空间中,所有进程都可以访问共享内存中的地址,就好像它们是由用C语言函数malloc分配的内存一样。而如果某个进程向共享内存写入数据,所做的改动将立即影响到可以访问同一段共享内存的任何其他进程。
特别提醒:共享内存并未提供同步机制,也就是说,在第一个进程结束对共享内存的写操作之前,并无自动机制可以阻止第二个进程开始对它进行读取。所以我们通常需要用其他的机制来同步对共享内存的访问,例如前面说到的信号量。
2、共享内存的原理
在Linux中,每个进程都有属于自己的进程控制块(PCB)和地址空间(Addr Space),并且都有一个与之对应的页表,负责将进程的虚拟地址与物理地址进行映射,通过内存管理单元(MMU)进行管理。两个不同的虚拟地址通过页表映射到物理空间的同一区域,它们所指向的这块区域即共享内存。
当两个进程通过页表将虚拟地址映射到物理地址时,在物理地址中有一块共同的内存区,即共享内存,这块内存可以被两个进程同时看到。这样当一个进程修改共享内存的数据时,另一个进程访问共享内存时就会得到新的数据。
3、面试题—>超过共享内存的大小限制怎么办?
在一个linux服务器上,共享内存的总体大小是有限制的,这个大小通过SHMMAX参数来定义(以字节为单位),您可以通过执行以下命令来确定 SHMMAX 的值:
# cat /proc/sys/kernel/shmmax
SHMMAX 的默认值是 32MB 。一般使用下列方法之一种将 SHMMAX 参数设为 2GB :
sysctl -w kernel.shmmax=2147483648
4、面试题—>同一个进程多次进行shmat会出现什么问题?
当首次创建共享内存段时,它并不能被任何进程所访问。为了使共享内存区可以被访问,则必须通过 shmat 函数将其附加( attach )到自己的进程空间中,这样进程就与共享内存建立了连接。
这样挂载一个共享内存如果是一次调用是没有问题的,但是一个进程是可以对同一个共享内存多次 shmat进行挂载的,物理内存是指向同一块,如果shmaddr为NULL,则每次返回的线性地址空间都不同。而且指向这块共享内存的引用计数会增加,也就是进程多块线性空间会指向同一块物理地址。这样,如果之前挂载过这块共享内存的进程的线性地址没有被shmdt掉,即申请的线性地址都没有释放,就会一直消耗进程的虚拟内存空间,很有可能会最后导致进程线性空间被使用完而导致下次shmat或者其他操作失败。
可以通过判断需要申请的共享内存指针是否为空来标识是否是第一次挂载共享内存,若是则使用进行挂载,若不是则退出。
void* ptr = NULL;
...
if (NULL != ptr)
return;
ptr = shmat(shmid,ptr,0666);
二、共享内存的使用
1. 调用函数shmget()创建一个新共享内存段或者取得一个既有的共享内存段的标识符;
2. 调用函数shmat()将共享内存附加到进程的虚拟地址空间中;
3. 为了引用共享内存,程序需要使用由shmat()函数返回的addr值,它是一个指向进程的虚拟地址空间中该共享内存段起点的指针;
4. 调用函数shmdt()分离共享内存段,调用之后,进程无法再引用这段共享内存。
5. 调用函数shmctl()删除共享内存段。只有一个进程需要执行这一步。
1、shmget函数
#include <sys/ipc.h>
#include <sys/shm.h>
int shmget(key_t key, size_t size, int shmflg);
- 功能:创建共享内存或者取得一个既有的共享内存段的标识符。
- 参数:
- key:程序需要提供一个参数key(非0整数),它有效地为共享内存段命名;
- size:size以字节为单位指定需要共享的内存容量,如果正在创建一个新段,则必须指定size。如果正在引用一个现存的段,则size指定为0.当创建一个新段,段内的内容初始化为0;
- shmflg:权限标志,它的作用与open函数的mode参数一样,如果要想在key标识的共享内存不存在时,创建它的话,可以与IPC_CREAT做或操作。共享内存的权限标志与文件的读写权限一样,举例来说,0644,它表示允许一个进程创建的共享内存被内存创建者所拥有的进程向共享内存读取和写入数据,同时其他用户创建的进程只能读取共享内存。
- 返回值:成功:返回一个与key相关的共享内存标识符(非负整数),用于后续的共享内存函数。失败:返回-1。
2、shmat函数
#include <sys/types.h>
#include <sys/shm.h>
void *shmat(int shm_id, const void *shm_addr, int shmflg);
- 功能:启动对该共享内存的访问,并把共享内存附加到当前进程的地址空间。
- 参数:
- shm_id:由shmget函数返回的共享内存标识;
- shm_addr:指定共享内存连接到当前进程中的地址位置,通常为空,表示让系统来选择共享内存的地址;
- shm_flg:一组标志位,通常为0。
- 返回值:成功:返回一个指向共享内存第一个字节的指针。失败:返回-1。
3、shmdt函数
#include <sys/types.h>
#include <sys/shm.h>
int shmdt(const void *shmaddr);
- 功能:将共享内存从当前进程中分离。注意,将共享内存分离并不是删除它,只是使该共享内存对当前进程不再可用。
- 参数:
- shmaddr:shmat函数返回的地址指针。
- 返回值:成功:返回0。失败:返回-1。
4、shmctl函数
#include <sys/shm.h>
int shmctl(int shm_id, int command, struct shmid_ds *buf);
- 功能:用来控制共享内存。
- 参数:
- shm_id:shmget函数返回的共享内存标识符;
- command:要采取的操作,它可以取下面的三个值 :
- IPC_STAT:把shmid_ds结构中的数据设置为共享内存的当前关联值,即用共享内存的当前关联值覆盖shmid_ds的值;
- IPC_SET:如果进程有足够的权限,就把共享内存的当前关联值设置为shmid_ds结构中给出的值;
- IPC_RMID:删除共享内存段。
- buf:一个结构指针,它指向共享内存模式和访问权限的结构。
struct shmid_ds
{
uid_t shm_perm.uid;
uid_t shm_perm.gid;
mode_t shm_perm.mode;
};
- 返回值:成功:返回0。失败:返回-1。
三、共享内存的demo
// ShmWrite.cpp
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/shm.h>
#include <sys/ipc.h>
#define BUFFER 4096
#define PERM S_IRUSR|S_IWUSR
int main(int argc, char *argv[])
{
// 创建一个新的共享内存段或者取得一个已有共享内存段的标识符
int shmid = shmget((key_t)123456, BUFFER,
PERM | IPC_CREAT | IPC_EXCL);
if (shmid == -1)
{
perror("shmget");
exit(1);
}
// 将共享内存段加载到调用进程的虚拟地址空间中
char* shmaddr = (char *)shmat(shmid, NULL, 0);
if (shmaddr == (void*)-1)
{
perror ("shmat");
exit(1);
}
// 从标准输入读数据
fgets(shmaddr, 100, stdin);
// 分离共享内存段
if (shmdt(shmaddr) == -1)
{
perror("shmdt");
exit(1);
}
sleep(10); // 睡眠十秒后共享内存失效
// 删除这块共享内存
if (shmctl(shmid, IPC_RMID, NULL) == -1)
{
perror("shmctl");
exit(1);
}
return 0;
}
// ShmRead.cpp
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>
#include <string.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/stat.h>
#include <sys/shm.h>
#define BUFFER 4096
#define PERM S_IRUSR|S_IWUSR
int main(int argc,char **argv)
{
int shmid = shmget((key_t)123456, BUFFER, PERM);
if (shmid == -1)
{
perror("shmget");
exit(1);
}
char* shmaddr = (char *)shmat(shmid, NULL, 0);
if (shmaddr == (void*)-1)
{
perror("shmat");
exit(1);
}
// 输出从另一个进程传过来的数据
printf("%s\n", shmaddr);
if (shmdt(shmaddr) == -1)
{
perror("shmdt");
exit(1);
}
return 0;
}
四、共享内存总结
共享内存就是映射一段能被其他进程所访问的内存,这段共享内存由一个进程创建,但多个进程都可以访问。共享内存是最快的IPC(进程间通信)方式,它是针对其它进程间通信方式运行效率低而专门设计的。它往往与其他通信机制,如信号量,配合使用,来实现进程间的同步与通信。
【优点】:
-
采用共享内存通信的一个显而易见的好处是效率高,因为进程可以直接读写内存,而不需要任何数据的拷贝。对于像管道和消息队列等通信方式,则需要在内核和用户空间进行四次的数据拷贝,而共享内存则只拷贝两次数据:一次从输入文件到共享内存区,另一次从共享内存区到输出文件。实际上,进程之间在共享内存时,并不总是读写少量数据后就解除映射,有新的通信时,再重新建立共享内存区域。而是保持共享区域,直到通信完毕为止,这样,数据内容一直保存在共享内存中,并没有写回文件。共享内存中的内容往往是在解除映射时才写回文件的。因此,采用共享内存的通信方式效率是非常高的。
-
不像匿名管道那样要求通信的进程有一定的父子关系,可用于任意两个进程之间通信。
【缺点】:
-
共享内存没有提供同步的机制,这使得我们在使用共享内存进行进程间通信时,往往要借助其他的手段来进行进程间的同步工作。
-
利用内存缓冲区直接交换信息,内存的实体存在于计算机中,只能同一个计算机系统中的诸多进程共享,不方便网络通信。
参考:https://www.cnblogs.com/wuchanming/p/4381910.html
https://blog.csdn.net/echo_ana/article/details/53456543