C++一分钟之-未来与承诺:std::future与std::promise

本文涉及的产品
大数据开发治理平台 DataWorks,不限时长
实时数仓Hologres,5000CU*H 100GB 3个月
实时计算 Flink 版,5000CU*H 3个月
简介: 【6月更文挑战第27天】`std::future`和`std::promise`是C++异步编程的关键工具,用于处理未完成任务的结果。`future`代表异步任务的结果容器,可阻塞等待或检查结果是否就绪;`promise`用于设置`future`的值,允许多线程间通信。常见问题包括异常安全、多重获取、线程同步和未检查状态。解决办法涉及智能指针管理、明确获取时机、确保线程安全以及检查未来状态。示例展示了使用`std::async`和`future`执行异步任务并获取结果。

在现代C++编程中,std::futurestd::promise是异步编程模型中的两个重要组件,它们构成了C++标准库中处理异步计算结果的基础。本文将深入浅出地介绍这两个概念,探讨它们的应用场景、常见问题、易错点及如何避免,同时辅以代码示例,帮助读者更好地理解和运用这些机制。
image.png

一、未来(std::future)与承诺(std::promise)

1.1 未来(std::future)

std::future代表一个可能尚未完成的异步任务的结果。一旦关联的任务完成,你可以通过future对象获取或等待这个结果。它就像是一个装着未来结果的容器,你可以选择阻塞等待结果,或者检查结果是否已准备好。

1.2 承诺(std::promise)

std::promise则是用来设置std::future值的对象。它允许你在某个时刻将结果存储起来,而这个结果可以被关联的future对象获取。promise就像是一个承诺,保证会提供一个结果给那些等待它的future

二、应用场景

  • 异步任务处理:当一个任务需要较长时间执行,且不希望阻塞主线程时,可以启动一个异步任务,并用std::future来接收其结果。
  • 并发编程:在多线程环境中,std::promisestd::future可以用来在不同线程间传递数据,实现线程间的通信。
  • 任务结果缓存:对于耗时但结果可复用的计算,可以先用std::async结合std::future执行一次,后续直接从future获取结果,避免重复计算。

三、常见问题与易错点

3.1 异常安全

当向std::promise设置值时抛出异常,如果没有妥善处理,可能会导致结果永远不会被设置,而等待的std::future将永远阻塞。

3.2 多重获取

std::future的结果只能获取一次。尝试再次调用get()会导致未定义行为。

3.3 错误的线程同步

在多线程环境下,没有正确同步对std::promise的访问可能导致数据竞争。

3.4 忘记检查std::future的状态

直接调用get()而不先检查is_ready()状态,可能会导致当前线程阻塞,特别是在结果还未准备好时。

四、如何避免这些问题

4.1 使用智能指针管理std::promise

利用std::shared_ptr<std::promise<T>>可以在异常发生时,通过智能指针的生命周期管理自动清理资源,确保结果能被正确设置。

4.2 明确获取结果的时机

使用std::future::wait_for()std::future::wait_until()来控制等待时间,避免无限期阻塞。

4.3 确保线程安全

使用互斥锁或其他同步原语保护对std::promise的操作,防止数据竞争。

4.4 检查未来状态

在调用get()之前,先检查std::future::valid()std::future::wait_for(),确保操作的安全性。

五、代码示例

下面的示例展示了如何使用std::async启动一个异步任务,并通过std::future获取结果。

#include <iostream>
#include <future>
#include <thread>
#include <chrono>

// 异步任务函数
int heavyComputation() {
   
   
    std::this_thread::sleep_for(std::chrono::seconds(2)); // 模拟耗时操作
    return 42; // 返回计算结果
}

int main() {
   
   
    // 启动异步任务并获取future
    std::future<int> result_future = std::async(std::launch::async, heavyComputation);

    std::cout << "Doing something else...\n";

    // 获取结果,如果结果还没准备好,这会阻塞直到结果可用
    int result = result_future.get();
    std::cout << "The result is: " << result << std::endl;

    return 0;
}

在这个例子中,heavyComputation函数在一个单独的线程中执行,而主线程继续执行其他任务,最后通过get()方法等待并获取结果。

通过理解std::futurestd::promise的工作原理及其最佳实践,开发者能够更高效、安全地编写异步和并发代码,充分利用现代硬件的多核优势,提升程序性能。

目录
相关文章
|
4天前
|
安全 C++
C++一分钟之-字符串处理:std::string
【6月更文挑战第25天】`std::string`是C++文本处理的核心,存在于`&lt;string&gt;`库中。它支持初始化、访问、连接、查找、替换等操作。常见问题包括空指针解引用、越界访问和不当内存管理。要安全使用,确保字符串初始化,用`at()`检查边界,用`.empty()`检查空字符串,且无需手动释放内存。高效技巧包括预先分配内存、利用互转函数以及使用迭代器。记得正确比较和遍历字符串以保证代码效率和安全性。
24 5
|
3天前
|
存储 设计模式 安全
C++一分钟之-并发编程基础:线程与std::thread
【6月更文挑战第26天】C++11的`std::thread`简化了多线程编程,允许并发执行任务以提升效率。文中介绍了创建线程的基本方法,包括使用函数和lambda表达式,并强调了数据竞争、线程生命周期管理及异常安全等关键问题。通过示例展示了如何用互斥锁避免数据竞争,还提及了线程属性定制、线程局部存储和同步工具。理解并发编程的挑战与解决方案是提升程序性能的关键。
23 3
|
16天前
|
C++
c++中的using namespace std;
c++中的using namespace std;
|
1月前
|
前端开发 Go
Golang深入浅出之-Go语言中的异步编程与Future/Promise模式
【5月更文挑战第3天】Go语言通过goroutines和channels实现异步编程,虽无内置Future/Promise,但可借助其特性模拟。本文探讨了如何使用channel实现Future模式,提供了异步获取URL内容长度的示例,并警示了Channel泄漏、错误处理和并发控制等常见问题。为避免这些问题,建议显式关闭channel、使用context.Context、并发控制机制及有效传播错误。理解并应用这些技巧能提升Go语言异步编程的效率和健壮性。
66 5
Golang深入浅出之-Go语言中的异步编程与Future/Promise模式
|
1月前
|
C++
【C++】std::string 转换成非const类型 char* 的三种方法记录
【C++】std::string 转换成非const类型 char* 的三种方法记录
28 0
|
1天前
|
C++
【C++】日期类Date(详解)②
- `-=`通过复用`+=`实现,`Date operator-(int day)`则通过创建副本并调用`-=`。 - 前置`++`和后置`++`同样使用重载,类似地,前置`--`和后置`--`也复用了`+=`和`-=1`。 - 比较运算符重载如`&gt;`, `==`, `&lt;`, `&lt;=`, `!=`,通常只需实现两个,其他可通过复合逻辑得出。 - `Date`减`Date`返回天数,通过迭代较小日期直到与较大日期相等,记录步数和符号。 ``` 这是236个字符的摘要,符合240字符以内的要求,涵盖了日期类中运算符重载的主要实现。
|
3天前
|
C++
C++职工管理系统(类继承、文件、指针操作、中文乱码解决)
C++职工管理系统(类继承、文件、指针操作、中文乱码解决)
7 0
C++职工管理系统(类继承、文件、指针操作、中文乱码解决)
|
1天前
|
存储 编译器 C++
【C++】类和对象④(再谈构造函数:初始化列表,隐式类型转换,缺省值
C++中的隐式类型转换在变量赋值和函数调用中常见,如`double`转`int`。取引用时,须用`const`以防修改临时变量,如`const int& b = a;`。类可以有隐式单参构造,使`A aa2 = 1;`合法,但`explicit`关键字可阻止这种转换。C++11起,成员变量可设默认值,如`int _b1 = 1;`。博客探讨构造函数、初始化列表及编译器优化,关注更多C++特性。
|
1天前
|
编译器 C++
【C++】类和对象④(类的默认成员函数:取地址及const取地址重载 )
本文探讨了C++中类的成员函数,特别是取地址及const取地址操作符重载,通常无需重载,但展示了如何自定义以适应特定需求。接着讨论了构造函数的重要性,尤其是使用初始化列表来高效地初始化类的成员,包括对象成员、引用和const成员。初始化列表确保在对象创建时正确赋值,并遵循特定的执行顺序。
|
1天前
|
C语言 C++
【C++】日期类Date(详解)③
该文介绍了C++中直接相减法计算两个日期之间差值的方法,包括确定max和min、按年计算天数、日期矫正及计算差值。同时,文章讲解了const成员函数,用于不修改类成员的函数,并给出了`GetMonthDay`和`CheckDate`的const版本。此外,讨论了流插入和流提取的重载,需在类外部定义以符合内置类型输入输出习惯,并介绍了友元机制,允许非成员函数访问类的私有成员。全文旨在深化对运算符重载、const成员和流操作的理解。

热门文章

最新文章