一、前言
除了标准的文件IO,例如open,read,write,内核还提供接口运行应用将文件map到内存,使得内存中的一个字节与文件中的一个字节一一对应。这就是今天要说的mmap
,它在android中的用处非常多,比如binder,还有腾讯的开源的IO框架MMKV。这里简单记录下使用。
二、优势和缺势
优势
1 .读写文件避免了
read()
和write()
系统调用,也避免了数据的拷贝。
2 .多个进程 map 同一个对象,可以共享数据。
3 .可以直接使用指针来跳转到文件某个位置,不必使用lseek()
系统调用。劣势
1 .内存浪费。由于必须要使用整数页的内存。
mmap原理
三、mmap使用方法
#include <sys/mman.h>
void * mmap (void *addr,
size_t len,
int prot,
int flags,
int fd,
off_t offset);
addr
这个参数是建议地址(hint),没有特别需求一般设为0。这个函数会返回一个实际 map 的地址。len
文件长度。prot
表明对这块内存的保护方式,不可与文件访问方式冲突。PROT_NONE
无权限,基本没有用PROT_READ
读权限PROT_WRITE
写权限PROT_EXEC
执行权限flags
描述了映射的类型。MAP_FIXED
开启这个选项,则 addr 参数指定的地址是作为必须而不是建议。如果由于空间不足等问题无法映射则调用失败。不建议使用。MAP_PRIVATE
表明这个映射不是共享的。文件使用 copy on write 机制映射,任何内存中的改动并不反映到文件之中。也不反映到其他映射了这个文件的进程之中。如果只需要读取某个文件而不改变文件内容,可以使用这种模式。MAP_SHARED
和其他进程共享这个文件。往内存中写入相当于往文件中写入。会影响映射了这个文件的其他进程。与MAP_PRIVATE
冲突。fd
文件描述符。进行 map 之后,文件的引用计数会增加。因此,我们可以在 map 结束后关闭 fd,进程仍然可以访问它。当我们 unmap 或者结束进程,引用计数会减少。offset
文件偏移,从文件起始算起。
如果失败,mmap 函数将返回 MAP_FAILED
。
四、Android中使用mmap
#include <jni.h>
#include <string>
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <android/log.h>
extern "C"
JNIEXPORT void JNICALL
Java_com_example_mmap_1demo_MainActivity_writeTestNative(JNIEnv *env, jobject thiz) {
std::string file = "/sdcard/mmap.txt";
//可读可写,没有就创建,unix打开
int m_fd = open(file.c_str(), O_RDWR|O_CREAT, S_IRWXU);
//获取当前一页大小
int m_size = getpagesize();
//将空文件设置成4k大小,4096字节
ftruncate(m_fd, m_size);
//产生映射 页内容可以写入可以读取 映射内容可以共享
int8_t * m_ptr = static_cast<int8_t *>(mmap(0, m_size, PROT_WRITE | PROT_READ, MAP_SHARED, m_fd,
0));
std::string data("这是一个mmap demo");
//内存拷贝
memcpy(m_ptr, data.data(), data.size());
__android_log_print(ANDROID_LOG_ERROR, "mmap", "写入数据:%s", data.c_str());
}
这个demo通过mmap文件映射到内存,只需要通过内存拷贝memcpy
向文件写入数据