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<uint8_t*>(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++免费学习笔记(深入)”;

Texta
Texta

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<mystruct></mystruct>),别边读边 push_back。每次扩容都会触发 memcpy,对千万级结构体就是灾难。

更优做法:

  • stat 获取文件大小,除以结构体大小,得到预估元素数量
  • reserve() 预分配容量,再用 resize() 或迭代器填充
  • 若结构体含指针或非 POD 类型,确保二进制布局兼容(加 static_assert(std::is_trivially_copyable_v<mystruct>)</mystruct>
  • 考虑用 std::span<uint8_t></uint8_t> 指向 mmap 区域,避免额外拷贝
struct Record { uint64_t id; float val; };
size_t file_size = /* ... */;
size_t count = file_size / sizeof(Record);
std::vector<Record> 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 不是银弹,它们把控制权交还给你,也把责任一并移交——对齐、同步、生命周期管理,漏掉任一环都可能引发静默错误或性能崩塌。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

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

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

470

2025.06.09

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

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

201

2025.07.04

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

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

743

2023.08.10

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

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

376

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

27

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

28

2026.01.21

C# 多线程与异步编程
C# 多线程与异步编程

本专题深入讲解 C# 中多线程与异步编程的核心概念与实战技巧,包括线程池管理、Task 类的使用、async/await 异步编程模式、并发控制与线程同步、死锁与竞态条件的解决方案。通过实际项目,帮助开发者掌握 如何在 C# 中构建高并发、低延迟的异步系统,提升应用性能和响应速度。

103

2026.02.06

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

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

1431

2023.07.26

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

23

2026.03.06

热门下载

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

精品课程

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

共48课时 | 10.3万人学习

Git 教程
Git 教程

共21课时 | 4.1万人学习

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

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