当前位置: 首页 > news >正文

C++11 异步编程(3)--- packaged_task

C++11 异步编程(3)— packaged_task

packaged_task 是 C++11 标准库中引入的一个非常实用的工具,主要用于将可调用对象(如函数、lambda 表达式、函数对象等)封装起来,以便在异步编程中方便地使用。

一、基本概念

packaged_task 是一个模板类,模板参数是可调用对象的调用签名。它将一个可调用对象封装起来,并且与一个 future 对象关联。当 packaged_task 被调用时,它会执行封装的可调用对象,并将结果存储在与之关联的 future 对象中。这样,调用者可以通过 future 对象获取来异步操作的结果。

二、构造函数

packaged_task 的构造函数用于初始化封装的可调用对象。以下是一些常见的构造方式:

2.1 直接构造

packaged_task<int()> task([](){return 42;})

这里创建另一个task对象,封装了一个返回42的lambda表达式。

2.2 通过移动构造函数

function<int()> f = [](){return 42;};
packaged_task<int()> task(std::move(f));

这里创建了一个function对象,然后通过移动构造函数将其移动到packaged_task对象中。

三、调用和执行

packaged_task 提供了多种方式来调用封装的可调用对象:

3.1直接调用

packaged_task<int()> task([](){return 42;});
future<int> fu =  task.get_future();
task();// 调用封装的可调用对象
cout << fu.get() << endl;

这里通过 task() 直接调用了封装的 lambda 表达式,并通过 future 对象获取了结果。

3.2 通过线程调用

#include <iostream>
#include<thread>
#include<future>
using namespace std;int Add(int a, int b)
{cout << "task is excute" << endl;return a + b;
}int main()
{// 创建一个 packaged_task 对象,封装 add 函数packaged_task<int(int, int)> task(Add);// 获取与 task 关联的 future 对象,用于获取任务结果future<int> future = task.get_future();// 在新线程中执行任务thread th(std::move(task),3,5);// 主线程可以继续执行其他操作std::cout << "Doing other things while the task is running..." << std::endl;// 获取异步任务的结果int nResult = future.get();cout << "the result of Task is = "<< nResult << endl;th.join();std::cout << "Hello World!\n";
}
  • 这里将 packaged_task 移动到线程中执行,线程结束后,可以通过 future 对象获取结果。

四、注意事项

线程安全packaged_task 本身不是线程安全的。如果需要在多线程环境中使用,需要自己进行同步控制。移动语义packaged_task 的对象不能被复制,只能被移动。这是因为它的内部资源(如与之关联的 future 对象)是唯一的,不能被复制。

异常处理:如果封装的可调用对象抛出异常,异常会被存储在与之关联的 future 对象中,可以通过 future 对象的 getwait 方法来获取异常。

五、使用场景

packaged_task 在异步编程中非常有用,尤其是在需要将任务提交到线程池、事件循环或其他异步执行环境中时。它提供了一种方便的方式来封装任务,并且能够通过 future 对象来获取任务的执行结果。

总的来说,packaged_task 是 C++11 引入的一个强大的工具,它简化了异步编程的复杂性,使得开发者能够更加方便地编写异步代码。

http://www.lqws.cn/news/569161.html

相关文章:

  • RDS MySQL vs. Aurora MySQL:高需求工作负载的终极迁移指南
  • 支持7种通信方式的通信测试工具
  • 面试150 有效的数独
  • 建造者模式 - Flutter中的乐高大师,优雅组装复杂UI组件!
  • TDengine 运维全攻略:五种备份与恢复方法深度解析(2025 最新版)
  • EPLAN Electric P8 2.9 零基础保姆级安装教程
  • 银行账户管理系统01
  • [Python] -基础篇3-掌握Python中的条件语句与循环
  • win上对调ctrl和alt键
  • java:如何用 JDBC 连接 TDSQL 数据库
  • HarmonyOS实战:自定义表情键盘
  • 云计算在布莱克-斯科尔斯模型中的应用:解析解、蒙特卡洛模拟与可视化-AI云计算数值分析和代码验证
  • FLOPS、FLOP/s、TOPS概念
  • Excel之证件照换底色3
  • Docker部署
  • 【Typst】纵向时间轴
  • 函数参数及数据结构说明
  • 一阶线性双曲型偏微分方程组的特征值与通解分析
  • ABP VNext + Twilio:全渠道通知服务(SMS/Email/WhatsApp)
  • RagFlow 更适合企业级深度应用,FastGPT 更适合快速开发和原型验证
  • 用户行为序列建模(篇十)-【加州大学圣地亚哥分校】SASRec
  • 对象的finalization机制Test
  • aws(学习笔记第四十八课) appsync-graphql-dynamodb
  • Java猜拳小游戏
  • 基于 SpringBoot 实现一个 JAVA 代理 HTTP / WS
  • node js入门,包含express,npm管理
  • SRS流媒体服务器之本地测试rtc推流bug
  • 2.安装Docker
  • 嵌入式硬件中电容的基本原理与详解
  • python动漫周边电商网站系统