共享内存的基本原理
1. 核心步骤
要在 Linux 中使用 System V 共享内存,通常遵循以下“四步走”:
- 创建/获取 (Create/Get):向内核申请一块共享内存,就像
malloc一样,但这是内核管理的。
- 系统调用:
shmget
- 系统调用:
- 关联 (Attach):把这块物理内存“挂接”到当前进程的虚拟地址空间(堆栈之间)。挂接后,这就变成了你进程内的一个指针。
- 系统调用:
shmat(Shared Memory Attach)
- 系统调用:
- 使用 (Use):像使用普通数组或指针一样读写数据。
- 去关联 (Detach):用完了,把这块内存从页表中去掉(断开联系)。
- 系统调用:
shmdt(Shared Memory Detach)
- 系统调用:
- 删除 (Control/Remove):如果所有进程都不用了,需要请求内核释放这块物理内存。
- 系统调用:
shmctl
- 系统调用:
2. 这里的“坑”:生命周期
这是一个极其重要的概念,也是和管道最大的区别。
- 管道:进程退出了,管道就自动销毁了(匿名管道),或者没人用时虽然文件在但数据清空了(命名管道)。
- 共享内存:随内核 (Kernel Persistence)。
- 如果你创建了共享内存,进程 A 退出了,进程 B 也退出了,但这块内存依然存在于内核中!
- 除非你显式调用代码删除它,或者重启操作系统,否则它会一直占用物理内存。
- 后果:调试代码时如果程序崩溃没来得及清理,你会发现内存泄漏,下次启动程序可能报错“File exists”。
系统调用详解
为了写代码,我们需要先认识这几个核心函数的参数。这部分比mkfifo复杂,涉及到一个新的概念:Key。
1.ftok—— 生成唯一标识符
怎么让两个无关的进程(Server 和 Client)找到同一个共享内存呢?
- 命名管道:靠文件路径(唯一)。
- 共享内存:靠Key 值(唯一)。
我们需要一个算法,把一个文件路径和一个项目 ID 转换成一个唯一的数字 Key。
#include <sys/types.h> #include <sys/ipc.h> key_t ftok(const char *pathname, int proj_id);pathname:一个存在的文件路径(通常用当前目录.)。proj_id:一个整数(通常写个字符,如'a')。- 返回值:成功返回生成的
key_t,两端只要传入相同的路径和 ID,就能得到相同的 Key。
2.shmget—— 创建共享内存
#include <sys/ipc.h> #include <sys/shm.h> int shmget(key_t key, size_t size, int shmflg);key:刚才ftok生成的那个 Key。size:申请多少字节?(建议是 4KB 的整数倍,因为操作系统按页分配内存,如果你申请 4097 字节,OS 实际会分配 8KB,但你只能用 4097)。shmflg:标志位和权限。
IPC_CREAT:不存在就创建,存在就获取。IPC_EXCL:配合IPC_CREAT使用,如果存在则报错(用于确保我是第一个创建的)。- 权限:类似文件权限,如
0666。
- 返回值:成功返回shmid(共享内存 ID),类似文件描述符
fd;失败返回 -1。
3.shmat—— 挂接
void *shmat(int shmid, const void *shmaddr, int shmflg);shmid:shmget返回的 ID。shmaddr:想挂载到虚拟内存的哪个地址?通常传NULL,让内核自己找个风水宝地。shmflg:读写模式,默认为 0(可读写)。- 返回值:成功返回共享内存的首地址(
void*),失败返回 -1。
4.shmctl—— 控制/删除
int shmctl(int shmid, int cmd, struct shmid_ds *buf);cmd:常用IPC_RMID(Remove ID),标记删除。- 注意:
IPC_RMID只是标记删除。只有当连接这块内存的所有进程都shmdt(断开)后,物理内存才会真正释放。