0

0

c++如何实现文件的异步读取_c++ std::future与std::launch::async使用【指南】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-03 02:49:35

|

978人浏览过

|

来源于php中文网

原创

C++标准库无真正的异步文件I/O;std::async只是将同步读操作扔进新线程,仍属阻塞式伪异步,适合CPU密集任务而非I/O密集场景。

c++如何实现文件的异步读取_c++ std::future与std::launch::async使用【指南】

C++ 标准库本身不提供真正的异步文件 I/O(比如类似 Linux 的 io_uring 或 Windows 的 OVERLAPPED);std::async + std::launch::async 只是把同步读取扔进独立线程,不是操作系统级异步。

为什么 std::async 读文件不是“真异步”

它只是用线程池或新线程调用 std::ifstream::read 这类阻塞函数,期间线程仍会挂起等待磁盘完成。CPU 不会并行处理 I/O 等待,只是把等待换到另一个线程里——对单文件意义不大,还可能因线程开销反而更慢。

  • 真正异步需要 OS 支持:libaio(Linux)、IOCP(Windows)、kqueuemacOS/BSD)
  • std::async 本质是“伪异步”,适合 CPU 密集型任务卸载,不适合 I/O 密集场景
  • 频繁小文件读 + std::async 容易触发线程创建/销毁开销,甚至耗尽线程资源

如何正确用 std::async 做“模拟异步读”

仅当你要避免主线程卡顿、且能接受线程切换成本时可用。关键点:显式管理生命周期、避免拷贝大缓冲区、检查异常。

auto read_task = std::async(std::launch::async, []() -> std::vector {
    std::ifstream file("data.bin", std::ios::binary);
    if (!file) throw std::runtime_error("cannot open file");
    
    file.seekg(0, std::ios::end);
    size_t size = file.tellg();
    file.seekg(0, std::ios::beg);
    
    std::vector buf(size);
    file.read(buf.data(), size);
    if (!file) throw std::runtime_error("read failed");
    
    return buf; // 移动语义自动生效
});

// 主线程可做其他事...
std::this_thread::sleep_for(10ms);

// 阻塞获取结果(此时才真正等待完成)
try {
    auto data = read_task.get(); // 注意:只可调用一次
} catch (const std::exception& e) {
    // 处理文件打开失败、读取失败等异常
}
  • 必须用 std::launch::async,否则 std::async 可能延迟执行(std::launch::deferred
  • 返回大对象(如 std::vector)时依赖移动语义,别写 return std::move(buf) —— 编译器会自动优化
  • read_task.get() 是唯一取结果方式,调用后 std::future 失效;重复调用抛 std::future_error
  • 异常在 get() 时重抛,不能在 lambda 内吞掉

替代方案:比 std::async 更实用的选择

除非你项目已重度依赖 std::thread 且不想引入第三方,否则以下方式更可靠:

Fotor AI Face Generator
Fotor AI Face Generator

Fotor 平台的在线 AI 头像生成器

下载

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

  • POSIX AIO(Linux/macOS):用 aio_read + aio_suspend,需链接 -lrt;但 API 繁琐、错误码分散、不支持 C++ RAII
  • Boost.ASIO:封装了跨平台异步文件 I/O(boost::asio::posix::stream_descriptor on Linux,boost::asio::windows::random_access_handle on Windows),支持 completion handler 和 co_await
  • C++20 协程 + 第三方库:如 liburing 绑定 + 自定义 awaiter,才能逼近零拷贝、无协程级异步
  • 简单场景直接用线程 + std::queue + std::condition_variable:比 std::async 更可控,避免 std::future 生命周期陷阱

真正要异步读文件,别卡在 std::async 上;它只是个线程启动器,不是 I/O 抽象层。系统调用的阻塞与否,不取决于你用什么 C++ 封装,而取决于你调的是 read() 还是 io_submit()

相关专题

更多
lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

202

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

189

2025.11.08

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

374

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

564

2023.08.10

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

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

473

2023.08.10

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

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

473

2023.08.10

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

54

2025.12.01

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

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

526

2023.07.26

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

150

2025.12.31

热门下载

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

精品课程

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

共48课时 | 6.4万人学习

Git 教程
Git 教程

共21课时 | 2.4万人学习

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

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