在终端下输入命令
ipcs -m
可以看到当前共享内存的信息,如下:
出现了全是key:0x00000000的共享内存,怎么回事呢?
首先我们自己先来建一个共享内存,代码如下:
#include <stdio.h>
#include <stdlib.h>
#include <sys/ipc.h>
#include <sys/shm.h>
int main()
{
key_t key = ftok(".", 1); //获取key
int shmid = shmget(key, 4, IPC_CREAT | 0666); //获取shmid
int *p_num = (int*)shmat(shmid, 0, 0); //挂接共享内存(映射)
if (p_num == (void*)-1) //出错给出信息并退出进程
{
perror("shmat");
exit(-1);
}
*p_num = 100; //往共享内存写入数据
while (1); //死循环,保持挂接
shmdt(p_num); //脱接共享内存(解除映射)
return 0;
}
再次 ipcs -m ,如下:
末尾多了自己创建的共享内存,接着我们用 ipcrm -m 命令来删除它
这下很清楚了吧,我们自己创建的共享内存的key变成了0x00000000,而status变成了dest,原因就是我们在程序中写了死循环,不让进程结束,只要挂接数(nattch)还不为0,说明共享内存还被占用,所以无法删除,但是以dest作为标记,表明只要进程结束,就会自动删除共享内存,接下来我们让程序结束,看是不是真的会删除,如下:
果然,我们创建的共享内存不见了。
ok,接下来只要知道以上共享内存被哪些进程占用,然后kill进程就行了,问题是我怎么知道是哪个进程在使用呢?
其实只要使用shmctl函数就可以获取到这个信息,最后我们以末尾的shmid:655376为例,来编写一个程序获取它的进程ID,结束它,如下:
#include <stdio.h>
#include <stdlib.h>
#include <sys/ipc.h>
#include <sys/shm.h>
int main()
{
int shmid = 655376; //定义shmid为655376
struct shmid_ds ds; //声明一个结构体类型的shmid_ds保存共享内存的信息
int res = shmctl(shmid, IPC_STAT, &ds); //查询共享内存
if (res == -1)
{
perror("shmctl");
exit(-1);
}
printf("cpid = %d, lpid = %d\n", ds.shm_cpid, ds.shm_lpid); //获取创建PID和最后使用的PID
return 0;
}
接下来如下操作即可:
如我们所料,shmid:655376被删除了