共享内存(Shared Memory)深度全解:Linux高性能IPC的核心机制与实战

共享内存是Linux下唯一无需内核数据拷贝 的IPC机制,通过映射同一物理内存到多进程虚拟地址空间实现"零距离"数据交换。但其"无同步"特性使其成为最危险也最强大的IPC工具

1:何为零拷贝?

内存映射机制(x86_64示例)

关键:两个进程的虚拟地址经MMU转换后指向同一物理页帧

硬件保障:CPU缓存一致性协议(MESI/MOESI)自动同步多核缓存

零拷贝验证:

cpp 复制代码
strace -e trace=memory ./producer 2>&1 | grep -E "shm|map"
# 仅出现 shmat/mmap 系统调用,无 read/write

2:双标准深度剖析(含源码级差异)

2.1 System V 共享内存(shmget/shmat)

内核数据结构(include/linux/shm.h)

cpp 复制代码
struct shmid_kernel {
    struct kern_ipc_perm shm_perm; // 权限、key、shmid
    size_t shm_segsz;              // 段大小(字节)
    time64_t shm_atim;             // 最后附加时间
    time64_t shm_dtim;             // 最后分离时间
    time64_t shm_ctim;             // 最后修改时间
    pid_t shm_cprid;               // 创建者PID
    pid_t shm_lprid;               // 最后操作者PID
    unsigned long __shm_nattch;    // 当前附加进程数 ★
    // ... 
};

生命周期管理:

cpp 复制代码
// 创建(仅首次调用需IPC_CREAT)
int shmid = shmget(0xABCDEF, 1<<20, IPC_CREAT | 0600); 

// 附加(返回进程内虚拟地址)
void *addr = shmat(shmid, NULL, 0); 

// 分离(减少__shm_nattch计数)
shmdt(addr); 

// 标记删除(当__shm_nattch==0时真正释放)
shmctl(shmid, IPC_RMID, NULL); 

POSIX 共享内存(shm_open/mmap)

文件系统视角

cpp 复制代码
ls -l /dev/shm/          # tmpfs挂载点(内存文件系统)
total 4
-rw------- 1 user user 1048576 Jun 10 10:00 /my_shm_obj
df -h /dev/shm           # 显示为"磁盘",实为内存
tmpfs      7.8G  1.0M  7.8G   1% /dev/shm

生命周期:

cpp 复制代码
shm_unlink("/my_shm"); // 立即删除文件名(/dev/shm中消失)
// 但已mmap的进程仍可访问!直到所有munmap后内核回收内存

共享内存与互斥锁:

cpp 复制代码
// 共享内存结构体
typedef struct {
    pthread_mutex_t lock;
    pthread_cond_t cond;
    int data_ready;
    char payload[4096];
} SharedRegion;

// 初始化(关键属性设置!)
pthread_mutexattr_t mattr;
pthread_mutexattr_init(&mattr);
pthread_mutexattr_setpshared(&mattr, PTHREAD_PROCESS_SHARED); // 进程间共享
pthread_mutexattr_setrobust(&mattr, PTHREAD_MUTEX_ROBUST);    // 崩溃恢复 ★
pthread_mutex_init(&shm->lock, &mattr);

// 使用(崩溃安全)
if (pthread_mutex_lock(&shm->lock) == EOWNERDEAD) {
    pthread_mutex_consistent(&shm->lock); // 标记状态一致
    // 恢复临界区数据...
}
// ... 操作共享数据 ...
pthread_mutex_unlock(&shm->lock);
相关推荐
AlfredZhao3 小时前
vi 删除指定范围的行,不用再反复按 dd
linux·vi
clint4565 小时前
C++进阶(1)——前景提要
c++
用户9718356334669 小时前
银河麒麟 KY10 申威(SW64) 安装 nginx-1.16.1-2.p01.ky10.sw_64.rpm 详细步骤
linux
夜悊9 小时前
C++代码示例:进制数简单生成工具
c++
郝学胜_神的一滴10 小时前
CMake 021: IF 条件判据详诠
c++·cmake
猪脚踏浪10 小时前
linux 拷贝文件或目录到指定的位置
linux
_wyt0011 天前
洛谷 B3930 [GESP202312 五级] 烹饪问题 题解
c++·gesp
大树881 天前
金刚石散热越强,管路越先见顶
大数据·运维·服务器·人工智能·ai
摇滚侠1 天前
Linux CentOS7 rpm 安装 MySQL 5.7
linux·运维·mysql
霸道流氓气质1 天前
领域驱动设计(DDD)在 Spring Boot 微服务中的实践指南
运维·spring boot·微服务