直接运行 ipcs -m 可列出所有 system v 共享内存段,含 key、shmid、owner、perms、bytes、nattch 等字段;它不依赖进程运行状态,仅读取内核 ipc 数据,但非 root 用户默认只能看到自己创建的段,需 sudo ipcs -ma 查全部。

怎么用 ipcs -m 快速列出所有共享内存段
直接运行 ipcs -m 就能看到当前系统中所有 System V 共享内存段,包括 key、shmid(共享内存 ID)、owner、perms(权限)、bytes(大小)、nattch(关联进程数)等字段。这是最轻量、最通用的查看方式,不依赖进程是否还在运行,只查内核维护的 IPC 对象。
常见错误现象:执行后返回空或提示 Permission denied——多数是因为非 root 用户看不到其他用户的共享内存段(默认行为)。若需查看全部,加 -a 参数:ipcs -ma;但普通用户仍可能被权限限制,此时需联系管理员或改用 sudo ipcs -ma。
-
key是 ftok() 生成的整数值,用于跨进程唯一标识同一块共享内存;值为0x00000000表示该段是“私有 key”,通常由IPC_PRIVATE创建,无法被其他进程通过shmget()获取 -
nattch为 0 不代表可安全删除——它只表示当前无进程挂接,但该段仍驻留在内核中,直到显式调用shmctl(..., IPC_RMID, ...)或系统重启 - 如果看到大量 shmid 存在且
nattch == 0,大概率是程序异常退出后没调用shmctl(..., IPC_RMID, ...),属于典型的“共享内存泄漏”
为什么 ipcs 看不到你代码里创建的共享内存
不是所有“共享内存”都走 System V 路线。如果你用的是 mmap() + /dev/shm(POSIX 共享内存),ipcs -m 完全不会显示——它只管 System V 的 shmget()/shmctl() 体系。
验证方法:检查你的代码是否调用了 shm_open() 或 mmap() 映射了 /dev/shm/xxx 文件。这种情况下,得换命令:ls -l /dev/shm/ 查文件,df -h /dev/shm 看整体占用,find /dev/shm -type f -exec ls -lh {} \; 查具体对象大小。
- Linux 内核 2.6+ 默认挂载
/dev/shm为 tmpfs,大小受shmmax和shmall参数限制,和 System V 的/proc/sys/kernel/shmmax是两套独立参数 - System V 共享内存没有文件路径概念,POSIX 的有;前者靠
key和shmid标识,后者靠名字字符串(如/myshm) - 嵌入式低版本内核(如 2.4 或某些定制 Android 内核)可能压根不支持 POSIX 共享内存,只能用 System V,此时
ipcs -m是唯一可靠入口
如何定位某个 shmid 对应哪个进程在用
ipcs -m 只显示 nattch 数字,不列具体 PID。要查谁还挂着某段内存,得结合 /proc 文件系统手动翻:
先拿到 shmid(比如 3080264),然后运行:grep -r "shmid=3080264" /proc/[0-9]*/maps 2>/dev/null | cut -d':' -f1 | xargs -I{} basename {}。这会输出正在映射该 shmid 的进程 PID。
- 更稳妥的方式是用
lsof:lsof | grep "shmid=3080264",但部分精简系统可能没装lsof - 注意:即使进程已 exit,只要没调
shmdt(),其/proc/PID/maps中仍可能残留记录(尤其僵尸进程);真正释放要等所有引用计数归零 +IPC_RMID - 如果
nattch > 0但查不到对应 PID,可能是进程已崩溃但内核尚未清理页表项,属罕见但真实存在的 race condition
删错共享内存怎么办?ipcrm -m 的风险点
ipcrm -m <shmid></shmid> 会立即触发 IPC_RMID,内核标记该段为“待销毁”,但**不会立刻回收物理内存**——只有当所有已挂接的进程都调用 shmdt() 后,内存才真正释放。所以删完立刻 ipcs -m 还能看到它,只是 status 列会变灰或显示 dest(取决于工具版本)。
- 误删正在被多进程使用的 shmid,不会导致立即 crash,但后续任一进程调用
shmdt()或进程退出时,该段才会被真正清理;若程序逻辑依赖该段长期存在(如守护进程间通信),删掉会导致后续shmget()失败 - 别用
ipcs -m | awk '{print $2}' | xargs -I{} ipcrm -m {}批量清空——很可能把别人正在用的段也干掉了 - 安全做法:先
ipcs -m看 owner 和 nattch,确认 owner 是你自己的用户、nattch == 0,再删;或者用shmctl(shmid, IPC_STAT, &buf)在代码里检查buf.shm_nattch是否为 0
共享内存不像文件,没有“打开/关闭”的强语义,它的生命周期由 key、shmid、引用计数、显式控制四者耦合决定。看得到不代表能动,删得掉不代表已释放,这是最容易被忽略的底层事实。










