前言:
这篇文章我们来讲讲Linux——mmap
mmap
介绍mmap
接口介绍mmap
使用示例个人简介:努力学习ing
个人专栏:Linux
CSDN主页 愚润求学
其他专栏:C++学习笔记,C语言入门基础,python入门基础,C++刷题专栏
通过 系统调用mmap
,程序可以高效地访问文件数据,而无需通过传统的 read
或 write
系统调用进行数据的复制
具体来说:
传统的read
和write
分为两步(下面以write
为例)
使用mmap
write
,也无用户态到内核态的数据拷贝注意:当内核页缓存修改好了,我们就认为改好了,我们不关心内核缓冲区到磁盘的刷新这一过程(复杂)
当我们建立好映射以后,就可以直接对“开辟”的空间进行操作(虚拟地址)
我们对虚拟地址的操作,都是直接修改映射的内存。
头文件:
函数原型
void *mmap(void *addr, size_t length, int prot, int flags,
int fd, off_t offset);
参数说明
addr
:建议的映射起始地址(通常设为 NULL
,由内核自动选择)length
:映射区域的长度(必须是页大小(4 KB)的整数倍,如 4096
字节)prot
:映射区的内存保护模式选项(用|
链接)
PROT_READ
:映射区可读PROT_WRITE
:映射区可写PROT_EXEC
:映射区可执行<=
文件打开权限flags
:映射类型(如 MAP_SHARED
或 MAP_PRIVATE
)
MAP_PRIVATE
:创建⼀个私有映射。对映射区域的修改不会反映到底层文件中。(即:修改仅对当前进程有效(写时复制,类似 fork
)MAP_SHARED
:创建⼀个共享映射。对映射区域的修改会反映到底层文件中(即:修改会同步到文件,其他进程可见)MAP_ANONYMOUS
:指定要创建⼀个匿名内存映射 fd
:文件描述符(匿名映射时设为 -1
)offset
文件偏移量(开始映射的位置相较于0位置处的偏移)(必须是页大小的整数倍)返回值
(void*) -1
或者 MAP_FAILED
(等效的)注意
0
的文件无法映射,需要先调整文件大小
ftruncate(fd, SIZE)
(会把文件的内容全部初始化成\0
)mmap
需要读取文件元数据(如大小):所以,即使你只需要写入权限,也需要在open
文件的时候赋予读权限函数原型
int munmap(void *addr, size_t length);
参数介绍
addr
:映射空间的起始地址length
:空间长度(大小)返回值
- 成功:0
- 错误:-1
(错误码会被设置)
#include
#include
#include
#include
#include
#include
#include
#define FILENAME "log.txt"
#define SIZE 1024
int main()
{
// 打开文件
int fd = open(FILENAME, O_RDWR | O_APPEND | O_CREAT, 0666);
if(fd < 0)
{
perror("open");
return 1;
}
// 调整文件大小
ftruncate(fd, SIZE);
// 建立映射
char* mmap_addr = (char*)mmap(nullptr, SIZE, PROT_WRITE, MAP_SHARED, fd, 0);
if(mmap_addr == MAP_FAILED)
{
perror("mmap");
return 2;
}
// 写入操作
for(int c = 'a', i = 0; c <= 'z'; c++, i++)
{
mmap_addr[i] = c;
}
// 取消映射
munmap(mmap_addr, SIZE);
// 关闭文件
close(fd);
std::cout << "写入映射完毕" << std::endl;
return 0;
}
#include
#include
#include
#include
#include
#include
#include
#define FILENAME "log.txt"
int main()
{
// 打开文件
int fd = open(FILENAME, O_RDONLY);
if(fd < 0)
{
perror("open");
return 1;
}
struct stat st; // struct stat 类型的结构体用于记录文件的属性
fstat(fd, &st); // 获得fd对应的文件的结构体
// 建立映射
char* mmap_addr = (char*)mmap(nullptr, st.st_size, PROT_READ, MAP_SHARED, fd, 0);
if(mmap_addr == MAP_FAILED)
{
perror("mmap");
return 2;
}
// 读取操作
std::cout << mmap_addr << std::endl;
// 取消映射
munmap(mmap_addr, st.st_size);
// 关闭文件
close(fd);
std::cout << "读取映射完毕" << std::endl;
return 0;
}
在malloc里,对应大块的内存通常是使用mmap
来分配的,而对应小块的内存,是用brk
来分配的
这里要再介绍一个flags
选项:MAP_ANONYMOUS
MAP_ANONYMOUS
:指定要创建⼀个匿名内存映射。MAP_ANONYMOUS
标志时, mmap
会分配⼀段不与任何⽂件相关联的内存区域(即这段内存没有⽂件作为其后端存储)。下面用mmap
简单模拟实现一下malloc
:
#include
#include
#include
#include
#include
#include
#include
#define SIZE 1024
void* MyMalloc(int size)
{
// 建立映射(匿名映射)
void* mmap_addr = mmap(nullptr, size, PROT_WRITE | PROT_READ, MAP_SHARED | MAP_ANONYMOUS, -1, 0);
if(mmap_addr == MAP_FAILED)
{
perror("mmap");
exit(EXIT_FAILURE);
}
return mmap_addr;
}
void Myfree(void* mmap_addr, int size)
{
if(munmap(mmap_addr, size) == -1)
{
perror("munmap");
exit(EXIT_FAILURE);
}
}
int main()
{
char* ptr = (char*)MyMalloc(SIZE);
// 写入操作
for(int c = 'a', i = 0; c <= 'z'; c++, i++)
{
ptr[i] = c;
}
std::cout << "写入后地址内容是:" << ptr << std::endl;
Myfree(ptr, SIZE);
return 0;
}
我的分享也就到此结束啦
要是我的分享也能对你的学习起到帮助,那简直是太酷啦!
若有不足,还请大家多多指正,我们一起学习交流!
公主,王子:点赞→收藏⭐→关注
感谢大家的观看和支持!祝大家都能得偿所愿,天天开心!!!