共享内存其实是不同进程之间通过访问同一块内存进行通信。
之前博客有写道共享内存的一些接口,今天用代码实践了一下,证实了一个概念,,,共享内存的生命周期跟随操作系统(破涕为笑)
写端进程代码
#include<stdio.h>
#include <unistd.h>
#include <sys/shm.h>
int main()
{
//创建共享内存
int shmid=shmget(IPC_CREAT,1024,IPC_CREAT|0664);
if(shmid<0)
{
perror("shmid");
return 0;
}
//成功返回内存的操作句柄
void *addr=shmat(shmid,NULL,0);
if(!addr)
{
perror("shmat");
return 0;
}
//往内存当中写入数据
int count=0;
while(1)
{
sleep(2);
++count;
snprintf((char *)addr,1024,"%s--%d","mawenjie",count);
}
//分离进程
shmdt(addr);
//销毁内存
shmctl(shmid,IPC_RMID,NULL);
return 0;
}
读取共享内存的进程代码
#include <stdio.h>
#include <unistd.h>
#include <sys/shm.h>
int main()
{
//创建共享内存
int shmid=shmget(IPC_CREAT,1024,IPC_CREAT|0664);
if(shmid<0)
{
perror("shmid");
return 0;
}
//成功返回内存的操作句柄
void *addr=shmat(shmid,NULL,0);
if(!addr)
{
perror("shmat");
return 0;
}
//往内存当中写入数据
int count=0;
while(1)
{
sleep(2);
++count;
printf("%s~~\n",(char *)addr);
}
//分离进程
shmdt(addr);
//销毁内存
shmctl(shmid,IPC_RMID,NULL);
return 0;
}
当关闭写端进程时,内存当中的数据依然可以读取,和管道不同的是共享内存读取数据并不会将数据拿走。
共享内存接口描述link
查看共享内存
指令:ipcs -m
key:共享内存的标识符
shmid:共享内存的操作句柄
owner:所属者
perms:权限
bytes:共享内存的大小
nattch:附加的进程数量
status:共享内存状态
删除共享内存
ipcrm -m shmid(共享内存的操作句柄)
当删除一个有进程附加的共享内存时,操作系统会将内存的标识符设置为0x00000000,表示当前的共享内存不能再被别的进程所附加,并且将内存状态设置为dest状态。这样就会导致访问非法内存,程序越界访问奔溃掉,当找个进程退出后,操作系统会将描述共享内存的结构体释放掉。