异步编程之std::future(一): 使用
目录
1.概述
2.std::future的基本用法
4.std::future的使用场景
5.总结
1.概述
在编程实践中,我们常常需要使用异步调用。通过异步调用,我们可以将一些耗时、阻塞的任务交给其他线程来执行,从而保证当前线程的快速响应能力。还有一些场景可以通过将一个任务,分成多个部分然后将这部分交给多个线程来进行并发执行,从而完成任务的快速计算执行,提高应用性能。这里就需要用到异步调用的概念。
异步调用,就是当前线程将一个任务交给另外一个线程来进行执行,当前线程会接着执行当前任务,不需要等待这个交付给其他线程执行的任务的结果,直到其在未来的某一个时刻需要使用这个任务执行结果数据的时候。
std::future 是 C++11 标准库中引入的一个非常重要的特性,它提供了一种机制来访问异步操作的结果。当你启动一个异步操作(比如,通过 std::async、std::packaged_task、std::promise 等)时,你可以立即获得一个 std::future 对象,这个对象将在异步操作完成时持有操作的结果。
2.std::future的基本用法
要理解 std::future 的用法,我们需要先了解几个相关的组件:
-
std::async:用于启动异步任务,并返回一个std::future对象。 -
std::promise:用于设置将来某个时刻的值,可以与std::future配合使用。 -
std::future:用于获取异步操作的结果。
1)使用 std::async 创建异步任务
std::async 是最简单、最直接的创建异步任务的方式。我们来看一个简单的例子:
#include <iostream>
#include <future>
#include <chrono>int main() {// 启动一个异步任务,计算某个复杂的结果std::future<int> result = std::async(std::launch::async, []() -> int {std::this_thread::sleep_for(std::chrono::seconds(2)); // 模拟耗时操作return 42;});std::cout << "Doing something else while waiting for the result..." << std::endl;// 等待并获取结果int value = result.get();std::cout << "Result: " << value << std::endl;return 0;
}
在这个例子中,我们使用 std::async 启动了一个异步任务,该任务在后台计算一个结果并返回一个 std::future 对象。我们可以继续执行其他操作,直到需要获取异步任务的结果时,调用 result.get() 阻塞等待并获取结果。
2)使用 std::promise 和 std::future
除了 std::async,我们还可以使用 std::promise 和 std::future 来实现更灵活的异步任务管理。std::promise 用于在某个时刻设置结果,而 std::future 用于在稍后获取该结果。来看一个具体的例子:
#include <iostream>
#include <future>
#include <thread>void compute(std::promise<int>& prom) {std::this_thread::sleep_for(std::chrono::seconds(2)); // 模拟耗时操作prom.set_value(42); // 设置结果
}int main() {std::promise<int> prom;std::future<int> fut = prom.get_future();std::thread t(compute, std::ref(prom));std::cout << "Doing something else while waiting for the result..." << std::endl;int value = fut.get();std::cout << "Result: " << value << std::endl;t.join();return 0;
}
在这个例子中,我们创建了一个 std::promise 对象,并通过 get_future() 获取与之关联的 std::future 对象。然后,我们启动一个线程,在该线程中执行耗时操作并设置结果。主线程可以继续执行其他操作,直到调用 fut.get() 阻塞等待并获取结果。
3)std::future的尝试等待
std::future 还提供了 wait_for() 和 wait_until() 方法,允许你尝试等待一段时间或直到某个时间点。这些方法不会永久阻塞线程,而是会在指定的时间或条件到达时返回。如:
#include <iostream>
#include <vector>
#include <future>
#include <thread>
#include <chrono>int async_task(int id) {std::this_thread::sleep_for(std::chrono::seconds(id));return id * id;
}int main() {std::vector<std::future<int>> futures;for (int i = 1; i <= 3; ++i) {futures.push_back(std::async(std::launch::async, async_task, i));}for (auto& fut : futures) {std::cout << "Result: " << fut.get() << std::endl;}return 0;
}
4)std::future的错误处理
如果异步操作中抛出了异常,并且该异常没有被捕获,那么当你调用 get() 方法时,这个异常会被重新抛出。因此,你需要准备好处理这些可能的异常。如:
#include <iostream>
#include <future>
#include <thread>void compute(std::promise<int>& prom) {try {std::this_thread::sleep_for(std::chrono::seconds(2)); // 模拟耗时操作throw std::runtime_error("Something went wrong"); // 抛出异常prom.set_value(42); // 设置结果} catch (...) {prom.set_exception(std::current_exception()); // 设置异常}
}int main() {std::promise<int> prom;std::future<int> fut = prom.get_future();std::thread t(compute, std::ref(prom));std::cout << "Doing something else while waiting for the result..." << std::endl;try {int value = fut.get();std::cout << "Result: " << value << std::endl;} catch (const std::exception& e) {std::cerr << "Caught exception: " << e.what() << std::endl;}t.join();return 0;
}
在这个例子中,我们在 compute 函数中抛出了一个异常,并在 std::promise 对象中设置该异常。主线程在调用 fut.get() 时,会捕获并处理这个异常。
注意事项
- 每个
std::future对象只能调用一次get()方法。- 一旦
std::future对象的get()方法被调用并返回了结果,或者std::future对象被移动,它就不能再用来获取结果了。- 如果异步操作抛出异常且该异常在
std::future对象被销毁前未被捕获,则程序将终止。因此,请确保在销毁std::future对象之前调用get()方法并适当处理异常。
3.使用 std::shared_future
有时候,我们希望多个线程可以共享一个 std::future 对象的结果。C++11 引入了 std::shared_future,它允许多个线程安全地访问同一个异步操作的结果。来看一个例子:
#include <iostream>
#include <future>
#include <thread>
#include <vector>void compute(std::shared_future<int> fut) {std::cout << "Thread got result: " << fut.get() << std::endl;
}int main() {std::promise<int> prom;std::shared_future<int> fut = prom.get_future().share();std::vector<std::thread> threads;for (int i = 0; i < 3; ++i) {threads.emplace_back(compute, fut);}std::this_thread::sleep_for(std::chrono::seconds(2));prom.set_value(42);for (auto& t : threads) {t.join();}return 0;
}
在这个例子中,我们使用 std::promise 创建了一个 std::shared_future 对象,并将其传递给多个线程。这些线程可以共享并安全地访问同一个异步操作的结果。
4.std::future的使用场景
- 并行计算:当你需要将计算任务分配给多个线程并行执行,并需要等待所有任务完成时。
- 异步IO:在进行耗时的IO操作时,可以使用异步方式执行,并在操作完成时继续处理。
- 性能优化:通过异步处理非关键路径上的操作,可以提高应用程序的响应性和吞吐量。
5.总结
std::future 是 C++ 中进行异步编程的重要工具,它简化了异步操作的启动、结果查询和异常处理。通过合理使用 std::future 和相关机制,可以编写出既高效又易于维护的异步代码。
相关文章:
异步编程之std::future(一): 使用
目录 1.概述 2.std::future的基本用法 3.使用 std::shared_future 4.std::future的使用场景 5.总结 1.概述 在编程实践中,我们常常需要使用异步调用。通过异步调用,我们可以将一些耗时、阻塞的任务交给其他线程来执行,从而保证当前线程的…...
Vue3 + JS项目配置ESLint Pretter
前言 如果在开发大型项目 同时为多人协作开发 那么 ESLint 在项目中极为重要 在使用 ESLint 的同时 也需要使用 Pretter插件 统一对代码进行格式化 二者相辅相成 缺一不可 1. 安装 VsCode 插件 在 VsCode 插件市场搜索安装 ESLint 和 Pretter 2. 安装依赖 这里直接在 pac…...
JavaScript (十四)——JavaScript typeof和类型转换
目录 JavaScript typeof, null, 和 undefined typeof 操作符 null undefined undefined 和 null 的区别 JavaScript 类型转换 JavaScript 数据类型 JavaScript 类型转换 将数字转换为字符串 将布尔值转换为字符串 将日期转换为字符串 将字符串转换为数字 一元运算符…...
CTF-web 基础
网络协议 OSI七层参考模型:一个标准的参考模型 物理层 网线,网线接口等。 数据链路层 可以处理物理层传入的信息。 网络层 比如IP地址 传输层 控制传输的内容的传输,在传输的过程中将要传输的信息分块传输完成之后再进行合并。 应用…...
CP AUTOSAR标准之ChineseV2XNetwork(AUTOSAR_SWS_ChineseV2XNetwork)(更新中……)
1 简介和功能概述 本文档指定了AUTOSAR基础软件模块中国车辆对接网络(CnV2xNet)的功能、API和配置。 中国车联网网络(CnV2xNet)与中国车联网消息(CnV2xMsg)、中国车联网管理(CnV2xMgt)、中国车联网安全(CnV2xSec)以及AUTOSAR BSW模块以太网接口(EthIf)共同构成了AUTOSAR架构…...
【hloc】 项目流程
hloc 项目流程 1. 数据集准备2. 特征提取3. 匹配特征4. 三维重建5. 定位6. 结果评估7. 示例脚本 这个项目涉及到了视觉定位和三维重建的一系列步骤,从特征提取、匹配、三维重建到定位和结果评估。通过提供的脚本文件,用户可以方便地运行整个流程。 1. 数…...
鸿蒙系统开发【应用接续】基本功能
应用接续 介绍 基于ArkTS扩展的声明式开发范式编程语言编写的一个分布式视频播放器,主要包括一个直播视频播放界面,实现视频播放时可以从一台设备迁移到另一台设备继续运行,来选择更合适的设备继续执行播放功能以及PAD视频播放时协同调用手…...
nextTick方法的作用是什么?什么时候会用到
nextTick 方法在 Vue.js 中扮演着重要的角色,它用于在下次 DOM 更新循环结束之后执行延迟回调。这主要用于确保在 Vue 完成 DOM 更新后执行依赖于 DOM 的操作。 作用 确保 DOM 更新完成:Vue 的 DOM 更新是异步的,当你修改了数据后࿰…...
多 NodeJS 环境管理
前言 对于某个项目依赖特定版本的 NodeJS,或几个项目的 NodeJS 版本冲突时,需要在系统中安装多个版本的 NodeJS,这时可以使用一些工具来进行多个 NodeJS 的管理。 有很多类似的 NodeJS 管理工具,如 nvm, nvs, n 等,接…...
解决网站被植入跳转木马病毒
概述 网站被植入跳转木马病毒是一种常见的安全威胁,它可能导致网站用户被重定向到恶意站点。本文将指导您如何检测、清除这类木马病毒以及采取预防措施。 步骤1:确认感染 首先,需要确认您的网站确实受到了跳转木马的影响。 示例ÿ…...
Node.js(6)——npm软件包管理
npm npm是Node.js标准的软件包管理器。 使用: 初始化清单文件:npm init-y(得到package.json文件,有则略过此命令)下载软件包:npm i 软件包名称使用软件包 示例: 初始状态下npm文件夹下只有server.js,下载软件包前看…...
区块链核心概念与技术架构简介
引言 区块链,一种分布式账本技术,不仅为数字货币提供了基础设施,更在金融、供应链、物联网等多个领域展现出广泛的应用前景。区块链技术被认为是继蒸汽机、电力、互联网之后,下一代颠覆性的核心技术。 如果说蒸汽机释放了人们的…...
≌图概念凸显包含射线V的直线W是比V长的线
黄小宁 x轴中:各非负数点xh≥0都变回自己即都作恒等变换,其余点x-h都变号为xh就使x轴失去负数点而变为射线V{xh≥0}。这x轴变为射线V⊂x轴是不保距变换即不是x轴的刚体运动使x轴不≌V⊂x轴(小学生都知道x轴不≌射线V)。据≌图概念…...
子路由的配置方法?
子路由的配置方法主要涉及到在Vue-router中定义嵌套路由,即一个路由内部包含多个子路由。以下是配置子路由的基本步骤: 1. 定义父路由 首先,在Vue Router中定义父路由。父路由可以像其他普通路由一样定义,但通常会有一个组件与之…...
【大模型从入门到精通2】openAI api的入门介绍2
互动对话界面的搭建 让我们来看看如何建立一个互动对话界面,用户可以在此输入查询,系统实时处理并显示响应。 import panel as pn # 用于构建图形用户界面# 初始化对话历史记录和GUI组件 conversation_history [] input_widget pn.widgets.TextInpu…...
【前端编程小白】的HTML从零入门到实战
之前有高中毕业生读了博客,想让我帮他找一些前端入门的内容,他们报的计算机专业,想利用开学前夕学习一下,我给他推荐了一些菜鸟教程呀什么的。后来想,看来还是很多人需要一些更加入门的可成的,而且很多教程…...
easyexcel读文件入批量入es
1. 封装实体类,并对应excel表中的列 Data public class User {private String md5;private String id; ExcelProperty(value "age")private String age;ExcelProperty(value "username")private String name;} 2. 批量入库 private void in…...
JS+H5打字练习器
实现功能 1.导入.TXT文件到打字练习内容(部分浏览器可能出于安全问题限制了这一步操作) 2.输入文本到打字练习内(弹出输入框,将要练习的内容输入至输入框) 3. 开始练习,并根据正误在打字练习内容文本上修…...
windows系统关闭开机自检硬盘
效果: 注册表关闭开机硬盘自检,你可以按照以下步骤操作: 打开注册表编辑器: 按 Win R 键打开“运行”对话框。输入 regedit 并按回车,打开注册表编辑器。 定位到自检相关的键: 依次展开以下路径&#x…...
【多线程开发 5】实践使用Lock和Condition
Lock和Condition Lock 线程之间同步或者竞争都需要锁这类结构,一般我们都会用Object的wait和signal搭配synchronized关键字进行多线程开发,但是很多时候会造成死锁的现象,这是因为synchroniezd无法破坏死锁的产生条件,但是Lock接…...
Python|GIF 解析与构建(5):手搓截屏和帧率控制
目录 Python|GIF 解析与构建(5):手搓截屏和帧率控制 一、引言 二、技术实现:手搓截屏模块 2.1 核心原理 2.2 代码解析:ScreenshotData类 2.2.1 截图函数:capture_screen 三、技术实现&…...
【Go】3、Go语言进阶与依赖管理
前言 本系列文章参考自稀土掘金上的 【字节内部课】公开课,做自我学习总结整理。 Go语言并发编程 Go语言原生支持并发编程,它的核心机制是 Goroutine 协程、Channel 通道,并基于CSP(Communicating Sequential Processes࿰…...
Mac下Android Studio扫描根目录卡死问题记录
环境信息 操作系统: macOS 15.5 (Apple M2芯片)Android Studio版本: Meerkat Feature Drop | 2024.3.2 Patch 1 (Build #AI-243.26053.27.2432.13536105, 2025年5月22日构建) 问题现象 在项目开发过程中,提示一个依赖外部头文件的cpp源文件需要同步,点…...
微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据
微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据 Power Query 具有大量专门帮助您清理和准备数据以供分析的功能。 您将了解如何简化复杂模型、更改数据类型、重命名对象和透视数据。 您还将了解如何分析列,以便知晓哪些列包含有价值的数据,…...
React---day11
14.4 react-redux第三方库 提供connect、thunk之类的函数 以获取一个banner数据为例子 store: 我们在使用异步的时候理应是要使用中间件的,但是configureStore 已经自动集成了 redux-thunk,注意action里面要返回函数 import { configureS…...
视觉slam十四讲实践部分记录——ch2、ch3
ch2 一、使用g++编译.cpp为可执行文件并运行(P30) g++ helloSLAM.cpp ./a.out运行 二、使用cmake编译 mkdir build cd build cmake .. makeCMakeCache.txt 文件仍然指向旧的目录。这表明在源代码目录中可能还存在旧的 CMakeCache.txt 文件,或者在构建过程中仍然引用了旧的路…...
20个超级好用的 CSS 动画库
分享 20 个最佳 CSS 动画库。 它们中的大多数将生成纯 CSS 代码,而不需要任何外部库。 1.Animate.css 一个开箱即用型的跨浏览器动画库,可供你在项目中使用。 2.Magic Animations CSS3 一组简单的动画,可以包含在你的网页或应用项目中。 3.An…...
解读《网络安全法》最新修订,把握网络安全新趋势
《网络安全法》自2017年施行以来,在维护网络空间安全方面发挥了重要作用。但随着网络环境的日益复杂,网络攻击、数据泄露等事件频发,现行法律已难以完全适应新的风险挑战。 2025年3月28日,国家网信办会同相关部门起草了《网络安全…...
[ACTF2020 新生赛]Include 1(php://filter伪协议)
题目 做法 启动靶机,点进去 点进去 查看URL,有 ?fileflag.php说明存在文件包含,原理是php://filter 协议 当它与包含函数结合时,php://filter流会被当作php文件执行。 用php://filter加编码,能让PHP把文件内容…...
tomcat入门
1 tomcat 是什么 apache开发的web服务器可以为java web程序提供运行环境tomcat是一款高效,稳定,易于使用的web服务器tomcathttp服务器Servlet服务器 2 tomcat 目录介绍 -bin #存放tomcat的脚本 -conf #存放tomcat的配置文件 ---catalina.policy #to…...
