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

相关推荐
Alice-YUE19 分钟前
【js高频八股】防抖与节流
开发语言·前端·javascript·笔记·学习·ecmascript
云泽80824 分钟前
C++11 核心特性全解:列表初始化、右值引用与移动语义实战
开发语言·c++
froginwe1142 分钟前
DOM 加载函数
开发语言
Hello eveybody1 小时前
介绍一下背包DP(Python)
开发语言·python·动态规划·dp·背包dp
AI进化营-智能译站1 小时前
ROS2 C++开发系列12-用多态与虚函数构建可扩展的ROS2机器人行为模块
开发语言·c++·ai·机器人
iCxhust1 小时前
微机原理实践教程(C语言篇)---A002流水灯
c语言·开发语言·单片机·嵌入式硬件·51单片机·课程设计·微机原理
Morwit1 小时前
QML组件之间的通信方案(暴露子组件)
c++·qt·职场和发展
qeen872 小时前
【数据结构】建堆的时间复杂度讨论与TOP-K问题
c语言·数据结构·c++·学习·
莎士比亚的文学花园2 小时前
Linux驱动开发(3)——设备树
开发语言·javascript·ecmascript
图码2 小时前
如何用多种方法判断字符串是否为回文?
开发语言·数据结构·c++·算法·阿里云·线性回归·数字雕刻