**在linux进程中有时需要进程之间相互的通信
mmap 内存映射就是其中一种 ,通过创建文件映射到磁盘,然后返回一个指针,这样就可以对磁盘进行读取
mmap()系统调用使得进程之间通过映射同一个普通文件实现共享内存。普通文件被映射到进程地址空间后,进程可以向访问普通内存一样对文件进行访问,不必再调用read(),write()等操作。
采用共享内存通信的一个显而易见的好处是效率高,因为进程可以直接读写内存,而不需要任何数据的拷贝。对于像管道和消息队列等通信方式,则需要在内核和用户空间进行四次的数据拷贝,而共享内存则只拷贝两次数据[1]:一次从输入文件到共享内存区,另一次从共享内存区到输出文件。实际上,进程之间在共享内存时,并不总是读写少量数据后就解除映射,有新的通信时,再重新建立共享内存区域。而是保持共享区域,直到通信完毕为止,这样,数据内容一直保存在共享内存中,并没有写回文件。共享内存中的内容往往是在解除映射时才写回文件的。因此,采用共享内存的通信方式效率是非常高的**
void *mmap(void *addr, size_t len, int prot, int flags, int fd, off_t offset);
参数addr:指定映射的起始地址,通常设为NULL,由内核来分配
参数length:代表将文件中映射到内存的部分的长度。
参数prot:映射区域的保护方式。可以为以下几种方式的组合:
PROT_EXEC 映射区域可被执行
PROT_READ 映射区域可被读取
PROT_WRITE 映射区域可被写入
PROT_NONE 映射区域不能存取
参数flags:映射区的特性标志位,常用的两个选项是:
MAP_SHARD:写入映射区的数据会复制回文件,且运行其他映射文件的进程共享
MAP_PRIVATE:对映射区的写入操作会产生一个映射区的复制,对此区域的修改不会写会原文件
参数fd:要映射到内存中的文件描述符,有open函数打开文件时返回的值。
参数offset:文件映射的偏移量,通常设置为0,代表从文件最前方开始对应,offset必须是分页大小的整数倍。
#include"string.h"
#include"sys/mman.h"
#include"stdlib.h"
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include"unistd.h"
int main()
{
int fd;
char *p=NULL;
int len;
fd=open("mytext.txt",O_RDWR|O_CREAT,0777); //注意对文件打开时的权限和mmap 映射区权限的关系
if(fd<0)
{
perror("open fail");
}
len=ftruncate(fd,4); //拓展文件大小 文件必须得有实际大小
p=mmap(NULL,4,PROT_READ|PROT_WRITE,MAP_SHARED,fd,0); //映射区的权限应该小于 文件打开的权限
if(p==MAP_FAILED)
{
perror("mmap error"); //当映射区 一旦创建成功 文件描述符关闭打开都不会对其有影响
}
//p++; 杜绝文件指针本社的++ 但可以定义一个 指针指向它 再操作
//否则在释放时候会报错
strcpy(p,"abc");
munmap(p,len);
close(fd);
return 0;
}
==父子进程之间独享0到3G的内存 父子进程共享文件描述符和mmap 映射区 ==