浅谈C++的future

std::future 是 C++ 标准库中的一个模板类,提供了一种机制来管理和获取异步任务的结果。它常与异步操作相关,允许你在不同线程中执行任务,并在将来(即"未来")某个时刻获取这些任务的结果。std::future 通常和 std::asyncstd::promisestd::packaged_task 等其他类一起使用,用于实现异步编程。

为什么需要 std::future

  1. 异步任务结果管理

    • 当你在一个新线程或异步操作中执行任务时,你希望能够在任务完成后获取它的结果。std::future 提供了一种安全的方式来获取这些结果。
    • thread.join() 不同,future 不会在任务运行时阻塞主线程,它允许主线程继续执行其他操作,直到任务完成。
  2. 任务完成时的同步

    • std::future 提供了阻塞和非阻塞的机制来同步任务。你可以调用 get() 来等待任务完成,也可以使用 wait_for()wait_until() 来检查任务是否已经完成。
  3. 防止数据竞争

    • std::future 的机制避免了线程间的数据共享问题。它通过提供一种类型安全的、线程安全的方式,将任务的结果传递回主线程,而不需要使用锁或其他同步原语来避免数据竞争。

std::future 的基本用法:

std::future 是通过异步调用(如 std::async)生成的。当任务开始执行时,std::future 用于获取该任务的返回值。下面是一个简单的示例,展示了如何使用 std::futurestd::async

cpp 复制代码
#include <iostream>
#include <future>
#include <chrono>
#include <thread>

// 一个简单的异步任务函数
int task() {
    std::this_thread::sleep_for(std::chrono::seconds(2));  // 模拟耗时任务
    return 42;
}

int main() {
    // 使用 std::async 启动异步任务,并返回 future 对象
    std::future<int> result = std::async(std::launch::async, task);

    // 主线程可以在此期间做其他事情
    std::cout << "Doing other work in main thread...\n";

    // 阻塞等待异步任务完成,并获取结果
    int value = result.get();  // 这里阻塞直到 task() 完成并返回结果
    std::cout << "Task result: " << value << std::endl;

    return 0;
}

解释:

  1. std::async :它启动一个异步任务(可以理解为在另一个线程中运行),并返回一个 std::future 对象。
  2. std::future :它包含了任务的返回值。在调用 result.get() 之前,std::future 中的结果是不可用的。get() 会阻塞当前线程,直到异步任务完成并返回结果。
  3. 任务的异步执行 :在调用 result.get() 之前,主线程不会被任务阻塞,可以继续执行其他操作。

std::future 的关键操作:

  1. get()

    • 阻塞当前线程,直到异步任务完成,并返回任务的结果。
    • 如果任务已经完成,get() 会立即返回结果。
    • 只能调用一次,调用后 std::future 将不再拥有结果。
  2. wait()

    • 阻塞当前线程,直到异步任务完成,但不会返回任务的结果。可以用于等待任务完成,但不需要立即获取结果。
  3. wait_for(duration)

    • 阻塞一段指定的时间(duration),如果任务在这段时间内完成,则返回 std::future_status::ready,否则返回 timeoutdeferred
  4. wait_until(time_point)

    • 等待直到某个时间点,如果任务在此之前完成,则返回 std::future_status::ready

std::future 的典型应用场景:

  1. 异步任务的返回值

    • 当你需要在另一个线程中执行一个任务,并且希望稍后获取这个任务的返回值,std::future 就是最好的选择。例如在下载文件、计算密集型任务等场景中,可以在后台线程执行任务,主线程继续其他操作,任务完成时再获取结果。
  2. 并行计算

    • 你可以使用多个 std::future 来并行运行多个任务,并在所有任务完成后再获取所有的结果。
  3. 异步回调替代

    • 传统的异步编程通常使用回调函数来处理任务的完成状态,但 std::future 提供了一种更加简洁和清晰的方式来管理异步任务,避免了回调地狱的问题。

为什么要有 std::future 这种属性的存在?

  1. 简化异步编程

    • std::future 使得异步编程更加简洁直观,你可以像处理普通返回值一样处理异步任务的结果,而不必编写复杂的同步代码。
  2. 避免共享数据竞争

    • std::future 提供了一种无需锁定和同步机制的安全方式来在线程间传递数据,避免了传统多线程编程中的数据竞争问题。
  3. 灵活的任务管理

    • 通过 wait()wait_for() 等方法,std::future 允许你灵活控制任务的执行和结果获取方式。例如,你可以选择阻塞等待任务,或者选择轮询任务状态以避免阻塞。

总结:

std::future 提供了一种便捷的方式来处理异步任务的结果。它允许在异步任务运行的同时,主线程继续执行其他操作,直到任务完成后再获取结果。这种机制避免了使用回调函数的复杂性和数据竞争问题,简化了异步编程模型。

相关推荐
weixin_30777913几秒前
Jenkins Branch API插件详解:多分支项目管理的核心引擎
java·运维·开发语言·架构·jenkins
@木辛梓4 分钟前
结构体 结构体c++
开发语言·c++
kyle~4 分钟前
虚拟仪器LabView(VI)
c++·python·ros·labview
小股虫6 分钟前
消息中间件关键技术、设计原理与实现架构总纲
java·开发语言·架构
洲星河ZXH8 分钟前
Java,日期时间API
java·开发语言·python
前端老曹9 分钟前
Jspreadsheet CE V5 使用手册(保姆版) 二
开发语言·前端·vue.js·学习
秋邱11 分钟前
AR 定位技术深度解析:从 GPS 到视觉 SLAM 的轻量化实现
开发语言·前端·网络·人工智能·python·html·ar
Boop_wu12 分钟前
[Java EE] 多线程进阶(3) [线程安全集合类]
开发语言·windows·python
雨疏风骤124015 分钟前
【FreeRTOS】任务、任务状态
开发语言·stm32·c#·rtos
SongYuLong的博客19 分钟前
开源 C 标准库(C Library)
c语言·开发语言·开源