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. 多进程并发需要信号量同步
相关推荐
水木流年追梦1 小时前
大模型入门-应用篇2-RAG (检索增强生成):从原理到 Python 实战
开发语言·python·算法·prompt
数智工坊1 小时前
【连续动作空间算法2】TD3:DDPG的终极进化,根治连续控制中的函数近似误差
论文阅读·人工智能·算法·计算机视觉
洛水水1 小时前
【力扣100题】25. 搜索二维矩阵 II
算法·leetcode·矩阵
样例过了就是过了1 小时前
LeetCode热题100 多数元素
c++·算法·leetcode·贪心算法
白羊by1 小时前
YOLOv8 官方损失函数详解(按任务分类)
人工智能·深度学习·算法·yolo·分类
沪漂阿龙1 小时前
面试题:逻辑回归是什么?为什么用 Sigmoid、对数损失、最大似然、Softmax、多分类、类别不平衡一文讲透
人工智能·算法·机器学习·分类·逻辑回归
WL_Aurora1 小时前
备战蓝桥杯国赛【Day 9】
python·算法·蓝桥杯
洛水水1 小时前
【力扣100题】20.合并 K 个升序链表
算法·leetcode·链表
闻缺陷则喜何志丹2 小时前
【动态规划 前缀和】P7074 [CSP-J2020] 方格取数|普及+
c++·算法·前缀和·动态规划·洛谷