linux查看系统中的共享内存:ipcs -m
删除共享内存:ipcrm -m 共享内存id号
共享内存(Shared Memory),指两个或多个进程共享一个给定的存储区。
特点:
1. 共享内存是最快的一种 IPC,因为进程是直接对内存进行存取。
2. 因为多个进程可以同时操作,所以需要进行同步。
3. 信号量+共享内存通常结合在一起使用,信号量用来同步对共享内存的访问。
原型:
#include <sys/ipc.h>
#include <sys/shm.h>
// 创建或获取一个共享内存:成功返回共享内存ID,失败返回-1
1 int shmget(key_t key, size_t size, int flag);
// 连接共享内存到当前进程的地址空间:成功返回指向共享内存的指针,失败返回-1
2 void *shmat(int shm_id, const void *addr, int flag);
// 断开与共享内存的连接:成功返回0,失败返回-1
3 int shmdt(void *addr);
// 控制共享内存的相关信息:成功返回0,失败返回-1
4int shmctl(int shm_id, int cmd, struct shmid_ds *buf);
当用shmget
函数创建一段共享内存时,必须指定其 size;而如果引用一个已存在的共享内存,则将 size 指定为0 。
当一段共享内存被创建以后,它并不能被任何进程访问。必须使用shmat
函数连接该共享内存到当前进程的地址空间,连接成功后把共享内存区对象映射到调用进程的地址空间,随后可像本地空间一样访问。
shmdt
函数是用来断开shmat
建立的连接的。注意,这并不是从系统中删除该共享内存,只是当前进程不能再访问该共享内存而已。
shmctl
函数可以对共享内存执行多种操作,根据参数 cmd 执行相应的操作。常用的是IPC_RMID
(从系统中删除该共享内存)。
代码示例:
shm_write.c
#include <stdio.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
int main()
{
key_t key;
int shmid;
char *shmaddr;
if((key = ftok(".",1)) < 0)//获取key值
{
perror("ftok");
exit(-1);
}
shmid = shmget(key,1024*4,IPC_CREAT|0666);//创建共享内存
if(shmid == -1)
{
perror("shmget");
exit(-1);
}
shmaddr = (char *)shmat(shmid,0,0);//连接共享内存
printf("shmat ok\n");
strcpy(shmaddr,"hello world");//写数据
sleep(5);
shmdt(shmaddr);//断开连接
shmctl(shmid,IPC_RMID,0);//删除共享内存
return 0;
}
shm_read.c
#include <stdio.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
int main()
{
key_t key;
int shmid;
char *shmaddr;
if((key = ftok(".",1)) < 0)
{
perror("ftok");
exit(-1);
}
shmid = shmget(key,1024*4,0);//获取共享内存
if(shmid == -1)
{
perror("shmget");
exit(-1);
}
shmaddr = (char *)shmat(shmid,0,0);//连接共享内存
printf("shmat ok\n");
printf("shmaddr:%s\n",shmaddr);
shmdt(shmaddr);
return 0;
}