共享内存的概念
共享内存是指被多个进程共享的一部分物理内存。共享内存是进程间共享数据的一种最快的方法,一个进程向共享内存的区域写了数据,共享这块内存区域的所有进程就可以立刻看到其中的内容了。
共享内存的实现可以分为两个部分:
一.创建共享内存,使用shmget函数;
二.映射共享内存,将这块创建的共享内存映射到具体的进程空间中去,使用shmat函数
创建共享内存
int shmget(key_t key, int size, int 参数shmflag)
key:标识共享内存的键值:0/IPC_PRIVATE。
当key的取值为IPC_PRIVATE,则函数shmget()将创建一块新的共享内存;如果key的取值为0,而参数shmflag中又设置IPC_PRIVATE这个标识,则同样会创建一块新的共享内存。
返回值:如果成功,返回共享内存标识符;如果失败,返回-1;
映射共享内存
int shmat(int shmid, char *shmaddr, int flag)
参数:
shmid:shmget函数返回的共享内存标识符
flag:决定什么方式来确定映射的地址(通常为0)
返回值:
如果成功,则返回共享内存映射到进程中的地址;如果失败,则返回-1;
注意:当一个进程不再需要共享内存时,需要把它从进程地址空间中脱离。
使用下列函数:int shmdt(char * shmaddr) 来解除。
示例运行
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <errno.h> //用于获取出错时的errno,供调试错误使用。
#include <unistd.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#define PERM S_IRUSR|S_IWUSR //设置共享内存权限
int main(int argc,char **argv)
{
int shmid;
char *p_addr,*c_addr;
if(argc!=2)
{
fprintf(stderr,"Usage:%s\n\a",argv[0]);
exit(1);
}
if((shmid=shmget(IPC_PRIVATE,1024,PERM))==-1) //依据设置的权值创建一块共享内存,返回共享内存标识符
{
fprintf(stderr,"Create Share Memory Error:%s\n\a",strerror(errno));
exit(1);
}
if(fork()) // 父进程写
{
p_addr=shmat(shmid,0,0); //在父进程中映射共享内存,返回值为共享内存映射到进程中的地址
//第二个参数为0,表示让操作系统自动为申请的空间开辟内存区域
memset(p_addr,'\0',1024); //共享内存初始化
strncpy(p_addr,argv[1],1024);//共享内存赋初值
wait(NULL); // 阻塞父进程,直到该父进程的子进程退出
exit(0);
}
else // 子进程读
{
sleep(1); // 如果子进程先运行的话,先使子进程阻塞,让父进程先运行
c_addr=shmat(shmid,0,0); //返回共享内存映射到该子进程中的地址
printf("Client get %s\n",c_addr);
exit(0);
}
}
运行结果:(./shmem 123456)
Client get 123456