在使用 mmap 进行文件映射时,数据的落盘通常涉及到以下几个函数:
msync():此函数用于将内存中的更改同步到文件。使用msync()可以确保你对映射区域所做的修改被写入底层文件中。
int msync(void *addr, size_t length, int flags);
munmap():解除映射后,系统会自动将数据同步到磁盘,但如果需要确保数据已完全写入,你可以在调用munmap()前先调用msync()。fsync()/fdatasync():如果你希望确保文件描述符对应的整个文件都已经被写入磁盘,可以使用这些函数。首先,需要通过open()获取文件描述符,然后在执行mmap()后,通过这些函数来强制将缓冲区中的数据写入磁盘。fsync(int fd)会同步文件的所有属性。fdatasync(int fd)则仅同步文件的数据部分(忽略元数据)。
示例代码
下面是一个简单示例,演示如何使用这些函数:
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/mman.h>
#include <unistd.h>
#include <string.h>
int main() {
int fd = open("example.txt", O_RDWR | O_CREAT, S_IRUSR | S_IWUSR);
if (fd == -1) {
perror("open");
exit(EXIT_FAILURE);
}
// Extend the file size
lseek(fd, sizeof("Hello, World!") - 1, SEEK_SET);
write(fd, "", 1);
char *mapped = mmap(NULL, sizeof("Hello, World!"), PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
if (mapped == MAP_FAILED) {
perror("mmap");
close(fd);
exit(EXIT_FAILURE);
}
// Modify the mapped memory
strcpy(mapped, "Hello, World!");
// Sync changes to disk
if (msync(mapped, sizeof("Hello, World!"), MS_SYNC) == -1) {
perror("msync");
munmap(mapped, sizeof("Hello, World!"));
close(fd);
exit(EXIT_FAILURE);
}
// Unmap and close the file descriptor
munmap(mapped, sizeof("Hello, World!"));
// Optional: Ensure data is flushed to disk at the file descriptor level.
fsync(fd);
close(fd);
return 0;
}
在这个示例中,我们打开一个文件并使用 mmap 将其映射到内存中,然后我们修改映射的内容,并通过调用 msync 确保更改被写回到底层文件。最后,我们还使用了 fsync 来确保所有更改都已经被实际写入磁盘。
内容由零声教学AI助手提供,问题来源于学员提问




