mmap如何运用

mmap是什么

mmap是C语言中的一个系统调用,它允许程序将一个文件或者其他的对象映射到进程的地址空间。这种机制使得进程可以直接读取和修改被映射的对象,而不需要通过I/O系统调用来访问。

mmap的主要用途包括:

* 将一个普通文件映射到内存中,以便进行高效的读写操作;

* 创建一个匿名内存映射,用于在进程之间共享数据;

* 在特定情况下,可以使用mmap来代替malloc等动态内存分配函数。

要使用mmap,你需要包含<sys/mman.h>头文件,并且调用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等值的组合;

flags参数指定映射的行为,可以是MAP_SHARED、MAP_PRIVATE等值的组合;

fd参数指定文件描述符,如果是匿名映射则设为-1;

offset参数指定从文件开头开始映射的位置。

当你完成映射后,可以通过munmap()函数来删除映射。

需要注意的是,由于mmap涉及到底层的内存管理和文件系统操作,因此使用不当可能会导致程序崩溃或者其他严重的问题。因此,在使用mmap之前,最好先了解其工作原理和使用注意事项。

mmap并发读写举例

以下是一个使用mmap在C语言中实现的并发读写示例:

#include <stdio.h>

#include <stdlib.h>

#include <sys/mman.h>

#include <pthread.h>

// 全局变量

int numThreads;

char* mappedFile;

// 读写函数

void* writeData(void* arg) {

long threadID = (long)arg;

for (int i = threadID; i < numThreads; i += numThreads) {

// 写入数据

mappedFile[i] = 'a' + i % ('z' - 'a');

}

}

void* readData(void* arg) {

long threadID = (long)arg;

for (int i = threadID; i < numThreads; i += numThreads) {

// 读取数据

char ch = mappedFile[i];

printf("%d: Read '%c'\n", threadID, ch);

}

}

int main(int argc, char** argv) {

// 解析命令行参数

if (argc != 2) {

printf("Usage: %s num_threads\n", argv[0]);

return 1;

}

numThreads = atoi(argv[1]);

// 打开一个文件

FILE* file = fopen("test.txt", "w+");

if (file == NULL) {

printf("Failed to open file.\n");

return -1;

}

// 设置文件大小

fseek(file, numThreads, SEEK_SET);

ftruncate(fileno(file), numThreads);

// 重新定位到文件开头

fseek(file, 0L, SEEK_SET);

// 映射整个文件

mappedFile = (char*) mmap(NULL, numThreads, PROT_READ | PROT_WRITE, MAP_SHARED, fileno(file), 0);

if (mappedFile == MAP_FAILED) {

printf("Failed to map file.\n");

return -1;

}

// 创建线程

pthread_t threads[numThreads];

for (int i = 0; i < numThreads; i++) {

if (i % 2 == 0) {

pthread_create(&threads[i], NULL, writeData, (void*)(long)i);

} else {

pthread_create(&threads[i], NULL, readData, (void*)(long)i);

}

}

// 等待所有线程完成

for (int i = 0; i < numThreads; i++) {

pthread_join(threads[i], NULL);

}

// 取消映射

munmap(mappedFile, numThreads);

return 0;

}

在这个示例中,首先解析命令行参数以确定线程数量。然后,创建一个足够大的空文件,并将其全部映射到内存中。接着,创建相同数量的线程,其中一半执行写操作,另一半执行读操作。最后,等待所有线程完成,并取消映射。

注意

在并发环境中使用mmap时,必须确保只有一个线程同时访问同一个映射区域。否则,可能会出现竞态条件或其他并发问题。本示例通过交替创建读写线程,避免了这个问题。

相关推荐
HABuo4 小时前
【linux文件系统】磁盘结构&文件系统详谈
linux·运维·服务器·c语言·c++·ubuntu·centos
2401_858936889 小时前
【Linux C 编程】标准 IO 详解与实战:从基础接口到文件操作实战
linux·c语言
季明洵12 小时前
C语言实现单链表
c语言·开发语言·数据结构·算法·链表
浅念-12 小时前
C语言编译与链接全流程:从源码到可执行程序的幕后之旅
c语言·开发语言·数据结构·经验分享·笔记·学习·算法
爱吃生蚝的于勒12 小时前
【Linux】进程信号之捕捉(三)
linux·运维·服务器·c语言·数据结构·c++·学习
The森12 小时前
Linux IO 模型纵深解析 01:从 Unix 传统到 Linux 内核的 IO 第一性原理
linux·服务器·c语言·经验分享·笔记·unix
C++ 老炮儿的技术栈13 小时前
Qt 编写 TcpClient 程序 详细步骤
c语言·开发语言·数据库·c++·qt·算法
wangjialelele14 小时前
Linux下的IO操作以及ext系列文件系统
linux·运维·服务器·c语言·c++·个人开发
wengqidaifeng16 小时前
数据结构(三)栈和队列(上)栈:计算机世界的“叠叠乐”
c语言·数据结构·数据库·链表
VekiSon17 小时前
Linux内核驱动——设备树原理与应用
linux·c语言·arm开发·嵌入式硬件