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