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

相关推荐
萝卜兽编程2 分钟前
优先级队列
c++·算法
Bruce小鬼3 分钟前
QT文件基本操作
开发语言·qt
2202_754421548 分钟前
生成MPSOC以及ZYNQ的启动文件BOOT.BIN的小软件
java·linux·开发语言
我只会发热15 分钟前
Java SE 与 Java EE:基础与进阶的探索之旅
java·开发语言·java-ee
懷淰メ25 分钟前
PyQt飞机大战游戏(附下载地址)
开发语言·python·qt·游戏·pyqt·游戏开发·pyqt5
hummhumm39 分钟前
第 22 章 - Go语言 测试与基准测试
java·大数据·开发语言·前端·python·golang·log4j
宁静@星空1 小时前
006-自定义枚举注解
java·开发语言
hummhumm1 小时前
第 28 章 - Go语言 Web 开发入门
java·开发语言·前端·python·sql·golang·前端框架
武子康1 小时前
Java-07 深入浅出 MyBatis - 一对多模型 SqlMapConfig 与 Mapper 详细讲解测试
java·开发语言·数据库·sql·mybatis·springboot
珹洺1 小时前
C语言数据结构——详细讲解 双链表
c语言·开发语言·网络·数据结构·c++·算法·leetcode