Linux 共享内存(shm)

共享内存(Shared Memory)是最快的进程间通信方式,因为它直接让多个进程访问同一块物理内存,无需数据拷贝。

我会用 C 语言 + System V 共享内存(shmget/shmat/shmdt/shmctl) 给你完整、可直接运行的实现方案,包含:

  1. 核心 API 讲解
  2. 写进程代码
  3. 读进程代码
  4. 运行与清理命令

一、共享内存核心 API

Linux 下共享内存标准接口:

c

运行

复制代码
// 1. 创建/获取共享内存段
int shmget(key_t key, size_t size, int shmflg);

// 2. 挂载共享内存到进程地址空间
void *shmat(int shmid, const void *shmaddr, int shmflg);

// 3. 卸载共享内存
int shmdt(const void *shmaddr);

// 4. 控制共享内存(删除、获取信息)
int shmctl(int shmid, int cmd, struct shmid_ds *buf);

二、完整实现:两个进程通信

1. 写进程(shm_write.c)

创建共享内存 → 写入数据

c

运行

复制代码
#include <stdio.h>
#include <stdlib.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <string.h>

// 约定的 KEY 值,两个进程必须相同
#define SHM_KEY 0x1234
// 共享内存大小
#define SHM_SIZE 1024

int main() {
    // 1. 创建共享内存
    int shmid = shmget(SHM_KEY, SHM_SIZE, IPC_CREAT | 0666);
    if (shmid == -1) {
        perror("shmget failed");
        exit(1);
    }
    printf("创建共享内存成功,ID:%d\n", shmid);

    // 2. 挂载到当前进程
    char *shmaddr = (char *)shmat(shmid, NULL, 0);
    if (shmaddr == (void *)-1) {
        perror("shmat failed");
        exit(1);
    }

    // 3. 写入数据
    const char *msg = "Hello, 共享内存测试!";
    strcpy(shmaddr, msg);
    printf("写入数据:%s\n", msg);

    // 4. 等待读进程读取
    printf("按回车退出...\n");
    getchar();

    // 5. 卸载共享内存
    shmdt(shmaddr);
    return 0;
}

2. 读进程(shm_read.c)

获取共享内存 → 读取数据

c

运行

复制代码
#include <stdio.h>
#include <stdlib.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <string.h>

#define SHM_KEY 0x1234
#define SHM_SIZE 1024

int main() {
    // 1. 获取已创建的共享内存
    int shmid = shmget(SHM_KEY, SHM_SIZE, 0666);
    if (shmid == -1) {
        perror("shmget failed");
        exit(1);
    }
    printf("获取共享内存成功,ID:%d\n", shmid);

    // 2. 挂载
    char *shmaddr = (char *)shmat(shmid, NULL, 0);
    if (shmaddr == (void *)-1) {
        perror("shmat failed");
        exit(1);
    }

    // 3. 读取并打印
    printf("读取数据:%s\n", shmaddr);

    // 4. 卸载
    shmdt(shmaddr);
    return 0;
}

三、编译与运行

1. 编译

bash

运行

复制代码
gcc shm_write.c -o shm_write
gcc shm_read.c -o shm_read

2. 运行(必须先启动写进程)

终端 1:

bash

运行

复制代码
./shm_write

终端 2:

bash

运行

复制代码
./shm_read

你会看到:

  • 写进程写入消息
  • 读进程直接读取到同一块内存的数据

四、共享内存清理(重要)

共享内存随内核存在,进程退出不会自动删除!

查看共享内存

bash

运行

复制代码
ipcs -m

删除指定共享内存

bash

运行

复制代码
ipcrm -m [shmid]

一次性删除所有共享内存

bash

运行

复制代码
ipcrm --all=shm

五、关键说明

  1. key 值
    • 两个进程必须使用相同的 key 才能找到同一块共享内存
  2. 生命周期
    • 内核持续管理,直到手动 ipcrm 删除或重启系统
  3. 同步问题
    • 共享内存不带同步机制,多进程同时读写会产生数据混乱
    • 实际使用需要配合:信号量 / 互斥锁

总结

  1. 共享内存是最快的进程通信方式
  2. 四步标准流程:shmgetshmat → 读写 → shmdt
  3. 必须手动用 ipcrm 清理内存
  4. 多进程并发需要信号量同步
相关推荐
林文韬3276 分钟前
逐位二进制拼接 → 翻转 → 去头零 → 消邻重
算法
变量未定义~6 分钟前
单点修改、区间求和(模板)、区间修改,单点查询(模板)
数据结构·算法
weixin_4684668516 分钟前
SURF 图像特征提取算法新手实战指南
图像处理·人工智能·算法·机器视觉·surf·sift
weixin_468466851 小时前
支持向量机新手实战指南
人工智能·python·算法·机器学习·支持向量机
weixin_468466852 小时前
机器学习之决策树新手实战指南
人工智能·python·算法·决策树·机器学习·ai
wanghu20242 小时前
ABC460_E题题解
c++·算法
z200509302 小时前
今日算法(回溯子集)
数据结构·算法·leetcode
Hesionberger2 小时前
巧用异或找出唯一数字(多解)
java·数据结构·python·算法·leetcode
变量未定义~2 小时前
阶乘的约数和、斐波那契数列、数列区间最大值(ST表)
数据结构·算法
智者知已应修善业2 小时前
【51单片机象棋快棋赛 电子裁判器】2023-12-27
c++·经验分享·笔记·算法·51单片机