0

0

C++读写大型二进制文件性能优化技巧【实战分享】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-18 07:43:20

|

846人浏览过

|

来源于php中文网

原创

mmap替代fread/fwrite可提升大文件随机读写性能,关键在于按需缺页加载、避免拷贝与上下文切换,但需注意msync同步、禁止realloc、对齐要求及多线程安全。

c++读写大型二进制文件性能优化技巧【实战分享】

mmap 替代 fread / fwrite 直接内存映射

对几百 MB 甚至 GB 级二进制文件做随机读写时,freadfwrite 的系统调用开销、缓冲区拷贝、用户态/内核态切换会明显拖慢速度。直接用 mmap(Linux/macOS)或 CreateFileMapping + MapViewOfFile(Windows)把文件“变成内存”,后续访问就像操作数组一样快。

关键点:

  • mmap 不会立即加载全部内容到物理内存,而是按需缺页加载,内存占用可控
  • 写入后需调用 msync(或 MAP_SYNC 标志,取决于内核版本)确保落盘,否则可能丢失数据
  • 避免在 mmap 区域内做 std::vector::push_back 或其他可能触发 realloc 的操作——指针会失效
  • 64 位程序下可安全映射超大文件;32 位下注意地址空间碎片,建议分段映射
int fd = open("data.bin", O_RDWR);
size_t file_size = lseek(fd, 0, SEEK_END);
void* addr = mmap(nullptr, file_size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
// 直接 reinterpret_cast(addr)[offset] = value;
msync(addr, file_size, MS_SYNC); // 确保写入磁盘
munmap(addr, file_size);
close(fd);

禁用 C 运行时缓冲,用 O_DIRECT 绕过页缓存(Linux)

当你要完全控制 I/O 路径(比如实现自定义缓存、避免与内核页缓存竞争),且文件大小远超可用内存时,O_DIRECT 可跳过内核页缓存,减少一次内存拷贝。但代价是:所有 read/write 必须满足对齐要求。

常见踩坑点:

立即学习C++免费学习笔记(深入)”;

Khroma
Khroma

AI调色盘生成工具

下载
  • 缓冲区地址必须按 getpagesize() 对齐(常用 posix_memalign 分配)
  • I/O 大小必须是页大小的整数倍(如 4096 字节
  • 文件偏移量也必须页对齐
  • O_DIRECT 在某些 SSD 或文件系统(如 ext4 with journaling)上可能反而更慢,务必实测
int fd = open("data.bin", O_RDWR | O_DIRECT);
void* buf;
posix_memalign(&buf, 4096, 4096);
ssize_t r = pread(fd, buf, 4096, 0); // offset=0 是页对齐的

批量处理 + 预分配 std::vector 避免频繁重分配

如果读取后要在内存中做结构化解析(比如把二进制流转成 std::vector),别边读边 push_back。每次扩容都会触发 memcpy,对千万级结构体就是灾难。

更优做法:

  • stat 获取文件大小,除以结构体大小,得到预估元素数量
  • reserve() 预分配容量,再用 resize() 或迭代器填充
  • 若结构体含指针或非 POD 类型,确保二进制布局兼容(加 static_assert(std::is_trivially_copyable_v)
  • 考虑用 std::span 指向 mmap 区域,避免额外拷贝
struct Record { uint64_t id; float val; };
size_t file_size = /* ... */;
size_t count = file_size / sizeof(Record);
std::vector records;
records.reserve(count);
records.resize(count); // 一次性分配+默认构造
// 然后用 fread 或 memcpy 填充 raw data

多线程读写要小心:文件偏移 vs 内存映射 vs 锁粒度

多个线程并发读写同一文件,最容易出错的是共享文件偏移(lseek + read 组合不是原子的)。即使用了 pread/pwrite,也要注意:

  • pread/pwrite 是线程安全的,但不解决数据竞争——多个线程往同一 offset 写,结果未定义
  • mmap 后,各线程直接操作内存地址,此时需用 std::atomic 或互斥锁保护临界区域,而非文件锁
  • 避免用 flockfcntl(F_SETLK) 控制大文件——锁粒度太粗,严重串行化
  • 真正高吞吐场景下,推荐按块划分:线程 A 处理 [0, 1GB),线程 B 处理 [1GB, 2GB),完全无共享状态

文件 I/O 性能瓶颈往往不在“怎么读”,而在“谁在读、读多少、是否重复拷贝”。mmap 和 O_DIRECT 不是银弹,它们把控制权交还给你,也把责任一并移交——对齐、同步、生命周期管理,漏掉任一环都可能引发静默错误或性能崩塌。

相关文章

数码产品性能查询
数码产品性能查询

该软件包括了市面上所有手机CPU,手机跑分情况,电脑CPU,电脑产品信息等等,方便需要大家查阅数码产品最新情况,了解产品特性,能够进行对比选择最具性价比的商品。

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

196

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

189

2025.07.04

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

481

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

143

2025.12.24

windows查看端口占用情况
windows查看端口占用情况

Windows端口可以认为是计算机与外界通讯交流的出入口。逻辑意义上的端口一般是指TCP/IP协议中的端口,端口号的范围从0到65535,比如用于浏览网页服务的80端口,用于FTP服务的21端口等等。怎么查看windows端口占用情况呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

578

2023.07.26

查看端口占用情况windows
查看端口占用情况windows

端口占用是指与端口关联的软件占用端口而使得其他应用程序无法使用这些端口,端口占用问题是计算机系统编程领域的一个常见问题,端口占用的根本原因可能是操作系统的一些错误,服务器也可能会出现端口占用问题。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1102

2023.07.27

windows照片无法显示
windows照片无法显示

当我们尝试打开一张图片时,可能会出现一个错误提示,提示说"Windows照片查看器无法显示此图片,因为计算机上的可用内存不足",本专题为大家提供windows照片无法显示相关的文章,帮助大家解决该问题。

791

2023.08.01

windows查看端口被占用的情况
windows查看端口被占用的情况

windows查看端口被占用的情况的方法:1、使用Windows自带的资源监视器;2、使用命令提示符查看端口信息;3、使用任务管理器查看占用端口的进程。本专题为大家提供windows查看端口被占用的情况的相关的文章、下载、课程内容,供大家免费下载体验。

452

2023.08.02

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

43

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 7.3万人学习

Git 教程
Git 教程

共21课时 | 2.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号