mmap
共享内存(Shared Memory):映射一段能被其他进程所访问的内存,这段共享内存由一个进程创建,但多个进程都可以访问
优点:
•无须复制,快捷,信息量大
缺点:
•通信是通过将共享空间缓冲区直接附加到进程的虚拟地址空间中来实现的,因此进程间的读写操作的同步问题
•利用内存缓冲区直接交换信息,内存的实体存在于计算机中,只能同一个计算机系统中的诸多进程共享,不方便网络通信
相关函数
创建映射区函数
void *mmap(void *addr, size_t length, int prot, int flags,int fd, off_t offset);
• addr NULL
•length 映射区的长度
• prot
○ PROT_READ 可读
○ PROT_WRITE 可写
•flags
○ MAP_SHARED 共享(对内存的修改,会影响到源文件)
○ MAP_PRIVATE 私有 (不会影响源文件)
•fd 文件描述符,open打开一个文件
• offset 偏移量
•返回值
○ 成功 返回 可用的内存首地址
○ 失败 返回 MAP_FAILED
释放映射区
int munmap(void *addr, size_t length);
• addr 传mmap的返回值
•length mmap创建的长度
•返回值 成功 0 失败-1
写端
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<sys/mman.h>
#include<string.h>
typedef struct _student{
int sid;
char sname[20];
}student;
int main(int argc, char *argv[]){
if(argc != 2){
printf("./a.out filename\n");
return -1;
}
int fd = open(argv[1],O_RDWR|O_CREAT|O_TRUNC,0666);
int length = sizeof(student);
ftruncate(fd,length);//指定文件的大小
student *stu = mmap(NULL,length,PROT_READ|PROT_WRITE,MAP_SHARED,fd,0);
if(stu == MAP_FAILED){
perror("mmap erro");
return -1;
}
int num = 0;
while(1){
stu->sid = num++;
sprintf(stu->sname,"wangdassye");
sleep(1);
}
munmap(stu,length);
close(fd);
return 0;
}
读端
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<sys/mman.h>
#include<fcntl.h>
#include<sys/wait.h>
typedef struct _student{
int sid;
char sname[20];
}student;
int main(int argc,char* argv[]){
int fd = open(argv[1],O_RDWR);
int length = sizeof(student);
student *stu = mmap(NULL,length,PROT_WRITE|PROT_READ,MAP_SHARED,fd,0);
if(stu == MAP_FAILED){
perror("mmap err");
return -1;
}
while(1){
printf("sid = %3d,sname = %s\n",stu->sid,stu->sname);
sleep(1);
}
munmap(stu,length);
close(fd);
return 0;
}
- 匿名映射 flag设置为 MAP_ANON 或者 MAP_ANONYMOUS(unix中没有这两个宏,可以用/dev/zero 做映射)。
- 用mmap支持无血缘关系进程通信:flag必须设置为MAP_SHARED