存储映射I/O使一个磁盘文件与存储空间中的一个缓冲区相映射。于是当从缓冲区中取数据,就相当于读文件中的相应字节。于此类似,将数据存入缓冲区,则相应的字节就自动写入文件。这样,就可在不适用read和write函数的情况下,使用地址(指针)完成I/O操作。
mmap()
是一个系统调用,用于在进程的虚拟地址空间中创建映射区域,从而将文件或其他设备映射到内存中。它允许进程直接访问这些映射区域,实现了文件和内存之间的无缝交互。
mmap()
函数的原型如下:
void *mmap(void *addr, size_t length, int prot, int flags, int fd, off_t offset);
下面是各个参数的解释:
addr
:指定映射区域所期望的起始地址,通常设为 NULL,让操作系统自动选择合适的地址。length
:映射区域的长度(以字节为单位)。prot
:指定映射区域的读写属性,可以使用PROT_READ
、PROT_WRITE
、PROT_EXEC
和PROT_NONE
进行位或运算。flags
:指定映射区域的类型和属性,可以使用MAP_SHARED
、MAP_PRIVATE
、MAP_FIXED
、MAP_ANONYMOUS
等进行位或运算。fd
:与映射区域关联的文件描述符。如果映射的是一个文件,需要指定对应的文件描述符,如果映射的是一个匿名映射区域,则为 -1。offset
:对于文件映射,指定从文件的哪个位置开始映射到内存。对于匿名映射区域,可以设置为 0。
mmap()
函数的返回值是映射区域的起始地址(通常是指向 void
类型的指针),如果映射失败,则返回 MAP_FAILED
。
以下是一个示例,展示了如何使用 mmap()
将文件映射到内存中:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include <pthread.h>
#include <fcntl.h>
#include <sys/mman.h>
void sys_err(const char* str)
{
perror(str);
exit(1);
}
int main(int argc, char* argv[])
{
char *p=NULL;
int fd;
fd=open("testmap.txt",O_RDWR|O_CREAT|O_TRUNC,0644);
if(fd==-1)
sys_err("open error");
ftruncate(fd,10);
int len =lseek(fd,0,SEEK_END);
p=mmap(NULL,len,PROT_READ|PROT_WRITE,MAP_SHARED,fd,0);
if(p == MAP_FAILED)
sys_err("mmap error");
strcpy(p,"hello mmap");
printf("-------------%s",p);
int ret = munmap(p,len);
if(ret == -1)
sys_err("munmap error");
return 0;
}
在上述示例中,我们首先使用 open()
函数打开一个文件,获取对应的文件描述符 fd
。然后,使用 fstat()
函数获取文件的大小。接下来,我们使用 mmap()
函数将文件映射到内存中,通过传递文件描述符 fd
和文件大小 file_size
来指定映射的文件和长度。如果映射成功,mmap()
函数将返回一个指向映射区域起始地址的指针 mapped_data
。你可以使用这个指针来直接访问映射的数据。
在映射操作完成后,我们可以使用 mapped_data
指向的内存进行读取操作。在示例中,我们没有展示具体的读取操作,但你可以使用指针来访问映射区域的数据。
最后,使用 munmap()
函数解除映射,并关闭文件描述符。
需要注意的是,mmap()
函数返回的映射区域指针通常是页对齐的,所以你可能需要进行适当的内存对齐处理。
此外,映射区域的访问权限和映射属性可以根据需要进行调整,具体取决于你的应用程序的需求和操作系统的支持。在实际使用 mmap()
函数时,请务必查阅相关文档和参考资料,并遵循正确的使用方法和最佳实践。