共享内存是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);