mmap
1 mmap原理
存储映射I/O (Memory-mapped I/O)使一个磁盘文件与存储空间中的一个缓冲区相映射。于是当从缓冲区中取数据,就相当于读文件中的相应字节。于此类似,将数据存入缓冲区,则相应的字节就自动写入文件。这样,就可在不适用read和write函数的情况下,使用地址(指针)完成I/O操作。使用存储映射这种方法,首先应通知内核,将一个指定文件映射到存储区域中。这个映射工作可以通过mmap函数来实现。
2 mmap的API
2.1 建立文件和内存的映射
#include <sys/mman.h>
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:私有的
MAP_ANONYMOUS:匿名映射,映射不受任何文件的支持,它的内容被初始化为零
fd和offset参数被忽略,如果指定MAP_ANONYMOUS,有些实现要求fd为-1
Linux从2.4内核开始支持MAP_ANONYMOUS与MAP_SHARED结合使用
fd:文件描述符,需要打开一个文件
offset:指定一个便宜位置,从该位置开始映射
返回值:
成功:返回映射区的首地址
失败:返回MAP_FAILED((void *)-1)
2.2 扩展文件大小
#include <unistd.h>
#include <sys/types.h>
int truncate(const char *path, off_t length);
参数:
path:要扩展的文件
length:要扩展的长度
2.3 释放映射区域
int munmap(void *addr, size_t length);
参数:
addr:映射区的首地址
length:映射区的长度
返回值:
成功:0
失败:-1
2.4 例:mmap的实用
2.4.1 write.c
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/mman.h>
#include <string.h>
int main(int argc, char const *argv[])
{
//通过open事先打开文件
int fd = open("tmp", O_RDWR | O_CREAT, 0666);
if(fd < 0)
{
perroe("open");
return 0;
}
//扩展文件大小
truncate("tmp", 16);
//建立映射
char *buf = (char *)mmap(NULL, 16, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0);
//使用区域
strcpy(buf, "hello mmap");
//断开映射
munmap(buf, 16);
return 0;
}
2.4.2 read.c
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/mman.h>
int main(int argc, char const *argv[])
{
//通过open事先打开文件
int fd = open("tmp", O_RDWR | O_CREAT, 0666);
if(fd < 0)
{
perroe("open");
return 0;
}
//扩展文件大小
truncate("tmp", 16);
//建立映射
char *buf = (char *)mmap(NULL, 16, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0);
//使用区域
printf("%s\n", buf);
//断开映射
munmap(buf, 16);
return 0;
}