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

C++笔记之std::async的用法

C++笔记之std::async的用法

code review!

文章目录

  • C++笔记之std::async的用法
    • 1.概念
    • 2.C++ 异步任务的使用示例 - 使用 std::async 和 std::future
    • 3. std::launch::async 和 std::launch::deferred
    • 4.如果需要真正的异步,请指定std::launch::async

1.概念

std::async 是 C++ 标准库中的一个函数,用于创建异步任务,允许在另一个线程中执行函数,并返回一个 std::future 对象,以便获取函数的结果。std::async 的一般用法如下:
在这里插入图片描述

#include <iostream>
#include <future>// 一个简单的函数,用于演示 std::async
int foo(int x, int y) {return x + y;
}int main() {// 使用 std::async 创建异步任务std::future<int> result = std::async(foo, 3, 4);// 等待异步任务完成并获取结果int sum = result.get();std::cout << "Sum: " << sum << std::endl;return 0;
}

上述示例中,std::async 接受一个函数(foo)和它的参数,并在后台创建一个新线程来执行该函数。函数执行完毕后,可以使用 std::future 对象的 get() 方法来获取函数的返回值。

以下是 std::async 的一些重要注意事项和用法:

  1. 返回值类型:std::async 返回一个 std::future 对象,可以用于获取异步任务的返回值。

  2. 默认策略:std::async 可以接受一个可选的策略参数,例如 std::launch::asyncstd::launch::deferred,用于指定任务是立即在新线程中执行还是延迟执行。如果不提供策略,默认行为由实现决定。

  3. 异常处理:如果异步任务抛出异常,std::future 对象的 get() 方法会重新抛出该异常。因此,需要适当地处理异常,或者使用 std::futurewait()wait_for() 方法来检查任务是否出现异常。

  4. 返回值的共享和移动语义:根据策略,std::async 可能共享或移动参数,因此要小心在异步任务中使用共享数据。

  5. 后台线程管理:std::async 会自动创建和管理后台线程,但不提供对线程的直接控制。如果需要更多的线程控制功能,可以考虑使用 std::thread

  6. 取消任务:C++标准库在std::async中不提供直接的取消任务的机制,因此需要通过其他方式来实现任务的取消。

2.C++ 异步任务的使用示例 - 使用 std::async 和 std::future

在这里插入图片描述

运行

在这里插入图片描述

代码

#include <iostream>
#include <future>int main() {// 使用std::async创建一个异步任务,并返回一个std::future对象std::future<int> future_result = std::async(std::launch::async, []() {// 模拟一个耗时操作,返回一个结果std::this_thread::sleep_for(std::chrono::seconds(2));return 42;});std::cout << "Waiting for the future to become ready..." << std::endl;// 使用std::future::get()等待任务完成并获取结果int result = future_result.get();std::cout << "Async task result: " << result << std::endl;return 0;
}

3. std::launch::async 和 std::launch::deferred

std::launch::asyncstd::launch::deferred 是用于控制 std::async 函数的执行策略的标志,它们决定了异步任务的行为。这两个标志可以作为 std::launch 枚举类型的成员来使用。以下是它们的详细解释:

  1. std::launch::async

    • 当使用 std::launch::async 标志调用 std::async 时,函数将在新的线程中立即执行,即使没有调用 std::future 对象的 get() 方法也会执行。
    • 这意味着任务会异步执行,不会阻塞当前线程,而是在后台创建一个新线程来执行任务。
    • 如果你希望任务尽快执行,并且可以与其他操作并发执行,可以选择此策略。
  2. std::launch::deferred

    • 当使用 std::launch::deferred 标志调用 std::async 时,函数不会立即执行,而是等待调用 std::future 对象的 get() 方法时再执行。
    • 这意味着任务会延迟执行,直到你真正需要结果为止。如果未调用 get(),任务可能根本不会执行。
    • 这种策略适用于需要懒惰地计算结果的情况,以节省计算资源。

默认情况下,如果不指定策略,std::async 的行为由 C++ 标准库实现决定,可以是 std::launch::async 也可以是 std::launch::deferred,取决于具体实现。因此,为了明确指定策略,你可以使用以下方式:

std::future<int> result = std::async(std::launch::async, foo, 3, 4); // 强制使用 std::launch::asyncstd::future<int> result = std::async(std::launch::deferred, foo, 3, 4); // 强制使用 std::launch::deferred

需要注意的是,使用 std::launch::async 策略可能会导致并发执行的线程数增加,而使用 std::launch::deferred 策略可能会导致任务延迟执行,因此需要根据具体情况选择适当的策略。

下面是一个完整的比较 std::launch::asyncstd::launch::deferred 策略的示例程序:

#include <iostream>
#include <future>int foo(int x, int y) {std::cout << "foo is running in thread " << std::this_thread::get_id() << std::endl;return x + y;
}int main() {// 使用 std::launch::async 策略,任务会立即在新线程中执行std::cout << "Using std::launch::async:" << std::endl;std::future<int> async_result = std::async(std::launch::async, foo, 3, 4);// 使用 std::launch::deferred 策略,任务会延迟执行,直到调用 get() 方法std::cout << "Using std::launch::deferred:" << std::endl;std::future<int> deferred_result = std::async(std::launch::deferred, foo, 3, 4);// 主线程继续执行其他工作std::this_thread::sleep_for(std::chrono::seconds(2));// 获取 std::launch::async 的结果(可能已经执行完毕)int async_sum = async_result.get();std::cout << "Result from std::launch::async: " << async_sum << std::endl;// 获取 std::launch::deferred 的结果(现在才执行)int deferred_sum = deferred_result.get();std::cout << "Result from std::launch::deferred: " << deferred_sum << std::endl;return 0;
}

在这个示例中,我们首先使用 std::launch::asyncstd::launch::deferred 策略调用了 foo 函数。foo 函数会输出当前线程的标识符。然后,主线程休眠了2秒钟以模拟其他工作。

接着,我们分别调用 async_result.get()deferred_result.get() 来获取两种策略下的结果。你会注意到,std::launch::async 的任务会立即执行(可能已经执行完毕),而 std::launch::deferred 的任务直到调用 get() 时才会执行。

这个示例演示了 std::launch::asyncstd::launch::deferred 的不同行为,以及如何使用它们来控制异步任务的执行方式。

4.如果需要真正的异步,请指定std::launch::async

在这里插入图片描述

相关文章:

C++笔记之std::async的用法

C笔记之std::async的用法 code review! 文章目录 C笔记之std::async的用法1.概念2.C 异步任务的使用示例 - 使用 std::async 和 std::future3. std::launch::async 和 std::launch::deferred4.如果需要真正的异步&#xff0c;请指定std::launch::async 1.概念 std::async 是 …...

OpenCV4(C++)—— 图像连通域的详细分析

文章目录 前言一、connectedComponents函数二、connectedComponentsWithStats函数 前言 图像连通域&#xff0c;其实就是图像分割的一种方法。它通过检测像素之间的连接关系和相似性来划分图像中的区域&#xff0c;以便进行后续处理。图像邻域和图像邻域分析就不介绍了&#x…...

Rule-Engine-Starter V1.0.0

一个轻量级的规则引擎、搜索引擎&#xff0c;让条件匹配简单、优雅。 GIT地址 https://gitcode.cosmoplat.com/15011240224/rule-engine-starter 介绍 Rule-Engine-Starter 是一个轻量级规则引擎&#xff0c;V1.0.0主要解决条件匹配问题。比如飞书文档&#xff0c;每个文档都…...

绘制X-Bar-S和X-Bar-R图,监测过程,计算CPK过程能力指数

X-Bar-S图和X-Bar-R图是统计质量控制中常用的两种控制图&#xff0c;用于监测过程的稳定性和一致性。它们的主要区别在于如何计算和呈现数据的变化以及所关注的问题类型。 X-Bar-S图&#xff08;平均值与标准偏差图&#xff09;&#xff1a; X-Bar代表样本均值&#xff0c;S代表…...

【每日一句】只出现一次的数

文章目录 Tag题目来源题目解读解题思路方法一&#xff1a;位运算 其他语言Cpython3 写在最后 Tag 【位运算-异或和】【数组】【2023-10-14】 题目来源 136. 只出现一次的数字 题目解读 给你一个数组&#xff0c;找出数组中只出现一次的元素。题目保证仅有一个元素出现一次&a…...

GDB调试程序常用命令

编译文件 g -g -o test test.cpp(注意&#xff1a;GDB调试的前提是在编译时加上-g参数.) 启动gdb # 方法一 gdb test # 方法二 gdb file test 设置断点 tbreak&#xff1a;设置临时断点&#xff0c;仅在第一次触发后失效。 watch&#xff1a;设置观察点&#xff0c;监控变量的…...

C语言,求两个数的二进制表达中,有多少个位数不同

以前我的博客中写过一篇求二进制的1的个数的博客&#xff0c;里面用按位与1的方式来判断位数是否为一。 如代码所示&#xff1a; #include <stdio.h> int num(int n) {int a 0;int i 0;while (i < 32){a a (n & 1);n n >> 1;i;}return a; } int main…...

解决Win10电脑无线网卡的移动热点无法开启问题

一、目的 利用无线网卡连接网络&#xff0c;然后又用无线网卡通过移动热点分享该网络。 移动热点&#xff0c;简单地说&#xff0c;就是将台式机或笔记本的 Internet 连接转化成 WIFI 信号以供移动设备无线上网的功能&#xff0c;硬件前提是电脑须安装有无线网卡。 二、问题 …...

Spring framework Day10:JSR330注入注解

前言 JSR330是Java社区标准化进程&#xff08;Java Community Process&#xff0c;简称JCP&#xff09;中的一个规范&#xff0c;全名为"Dependency Injection for Java"&#xff0c;即Java的依赖注入规范。它定义了一组注解和相关的规范&#xff0c;用于实现依赖注…...

Java开发中List数据量大,需要分片批次处理

在开发过程中可能会遇到需要处理的List数据量过大&#xff0c;可以选择分批处理的方式对大量数据进行处理。 1、使用 apache 的工具包 <dependency><groupId>org.apache.commons</groupId><artifactId>commons-collections4</artifactId><v…...

Apache Doris 2.0.2 版本正式发布!

峰会官网已上线&#xff0c;最新议程请关注&#xff1a;doris-summit.org.cn 点击报名 亲爱的社区小伙伴们&#xff0c;Apache Doris 2.0.2 版本已于 2023 年 10 月 6 日正式发布&#xff0c;该版本对多个功能进行了更新优化&#xff0c;旨在更好地满足用户的需求。有 92 位贡献…...

transformers架构实现

目录 架构代码如下 模型打印如下 架构代码如下 import numpy as np from torch.autograd import Variable import copy from torch import softmax import math import torch import torch.nn.functional as F import torch.nn as nn # 构建Embedding类来实现文本嵌入层 class…...

C++类型推导

这里对C的类型推导方式进行一次全面的总结。 C中有三种类型推导的方式&#xff0c;分别是模板、auto以及decltype()。以下分别介绍这三种方式的同异。 一 模板 假设有这样的函数模板和这样的调用&#xff1a; template<typename T> void f(ParamType param);f(expr);…...

Open3D(C++) SVD分解求两个点云的变换矩阵

目录 一、算法原理二、代码实现三、结果展示四、相关链接一、算法原理 计算两个点云的质心计算中心化向量计算协方差矩阵奇异值分解,求解旋转矩阵 R R R计算平移向量 t t...

rtmp htttp推流Windows桌面到srs进行播放

推流命令: ffmpeg -f gdigrab -framerate 30 -i desktop -c:v libx264 -preset ultrafast -tune zerolatency -pix_fmt yuv420p -f flv rtmp://xxx.xxx.xxxx.xx/live/livestream 后面是推流地址 推流后的播放地址为: http://xxxxxx:8080/live/livestream.flv 可以写一个…...

NSSCTF做题(9)

[GDOUCTF 2023]<ez_ze> 看见输入框而且有提示说是ssti注入 输入{{7*7}} 试试&#xff0c;发现报错 输入{%%}发现了是jinja2模板 找到关键函数 Python SSTI利用jinja过滤器进行Bypass ph0ebuss Blog 原理见这篇文章&#xff0c;这里直接给出payload {%set ninedict(aaa…...

【09】基础知识:React组件的生命周期

组件从创建到死亡它会经历一些特定的阶段。 React 组件中包含一系列勾子函数&#xff08;生命周期回调函数 <> 生命周期钩子函数 <> 生命周期函数 <> 生命周期钩子&#xff09;&#xff0c;会在特定的时刻调用。 我们在定义组件时&#xff0c;会在特定的生…...

Pytorch之ConvNeXt图像分类

文章目录 前言一、ConvNeXt设计决策1.设计方案2.Training Techniques3.Macro Design&#x1f947;Changing stage compute ratio&#x1f948;Change stem to "Patchify" 4.ResNeXt-ify5. Inverted Bottleneck6.Large Kernel Size7.Micro Design✨Replacing ReLU wit…...

Linux系统编程:makefile以及文件系统编程

增量编译概念 首先回顾一下我们之前写的各种gcc指令用来执行程序&#xff1a; 可以看见非常繁琐&#xff0c;两个文件就要写这么多&#xff0c;那要是成百上千岂不完蛋。 所以为了简化工作量&#xff0c;很自然的想到了将这些命令放在一起使用脚本文件来一键执行&#xff0c…...

《动手学深度学习 Pytorch版》 8.5 循环神经网络的从零开始实现

%matplotlib inline import math import torch from torch import nn from torch.nn import functional as F from d2l import torch as d2lbatch_size, num_steps 32, 35 train_iter, vocab d2l.load_data_time_machine(batch_size, num_steps) # 仍然使用时间机器数据集8.…...

FFmpeg 低延迟同屏方案

引言 在实时互动需求激增的当下&#xff0c;无论是在线教育中的师生同屏演示、远程办公的屏幕共享协作&#xff0c;还是游戏直播的画面实时传输&#xff0c;低延迟同屏已成为保障用户体验的核心指标。FFmpeg 作为一款功能强大的多媒体框架&#xff0c;凭借其灵活的编解码、数据…...

Go 语言接口详解

Go 语言接口详解 核心概念 接口定义 在 Go 语言中&#xff0c;接口是一种抽象类型&#xff0c;它定义了一组方法的集合&#xff1a; // 定义接口 type Shape interface {Area() float64Perimeter() float64 } 接口实现 Go 接口的实现是隐式的&#xff1a; // 矩形结构体…...

爬虫基础学习day2

# 爬虫设计领域 工商&#xff1a;企查查、天眼查短视频&#xff1a;抖音、快手、西瓜 ---> 飞瓜电商&#xff1a;京东、淘宝、聚美优品、亚马逊 ---> 分析店铺经营决策标题、排名航空&#xff1a;抓取所有航空公司价格 ---> 去哪儿自媒体&#xff1a;采集自媒体数据进…...

ip子接口配置及删除

配置永久生效的子接口&#xff0c;2个IP 都可以登录你这一台服务器。重启不失效。 永久的 [应用] vi /etc/sysconfig/network-scripts/ifcfg-eth0修改文件内内容 TYPE"Ethernet" BOOTPROTO"none" NAME"eth0" DEVICE"eth0" ONBOOT&q…...

LINUX 69 FTP 客服管理系统 man 5 /etc/vsftpd/vsftpd.conf

FTP 客服管理系统 实现kefu123登录&#xff0c;不允许匿名访问&#xff0c;kefu只能访问/data/kefu目录&#xff0c;不能查看其他目录 创建账号密码 useradd kefu echo 123|passwd -stdin kefu [rootcode caozx26420]# echo 123|passwd --stdin kefu 更改用户 kefu 的密码…...

免费PDF转图片工具

免费PDF转图片工具 一款简单易用的PDF转图片工具&#xff0c;可以将PDF文件快速转换为高质量PNG图片。无需安装复杂的软件&#xff0c;也不需要在线上传文件&#xff0c;保护您的隐私。 工具截图 主要特点 &#x1f680; 快速转换&#xff1a;本地转换&#xff0c;无需等待上…...

【从零学习JVM|第三篇】类的生命周期(高频面试题)

前言&#xff1a; 在Java编程中&#xff0c;类的生命周期是指类从被加载到内存中开始&#xff0c;到被卸载出内存为止的整个过程。了解类的生命周期对于理解Java程序的运行机制以及性能优化非常重要。本文会深入探寻类的生命周期&#xff0c;让读者对此有深刻印象。 目录 ​…...

计算机基础知识解析:从应用到架构的全面拆解

目录 前言 1、 计算机的应用领域&#xff1a;无处不在的数字助手 2、 计算机的进化史&#xff1a;从算盘到量子计算 3、计算机的分类&#xff1a;不止 “台式机和笔记本” 4、计算机的组件&#xff1a;硬件与软件的协同 4.1 硬件&#xff1a;五大核心部件 4.2 软件&#…...

苹果AI眼镜:从“工具”到“社交姿态”的范式革命——重新定义AI交互入口的未来机会

在2025年的AI硬件浪潮中,苹果AI眼镜(Apple Glasses)正在引发一场关于“人机交互形态”的深度思考。它并非简单地替代AirPods或Apple Watch,而是开辟了一个全新的、日常可接受的AI入口。其核心价值不在于功能的堆叠,而在于如何通过形态设计打破社交壁垒,成为用户“全天佩戴…...

wpf在image控件上快速显示内存图像

wpf在image控件上快速显示内存图像https://www.cnblogs.com/haodafeng/p/10431387.html 如果你在寻找能够快速在image控件刷新大图像&#xff08;比如分辨率3000*3000的图像&#xff09;的办法&#xff0c;尤其是想把内存中的裸数据&#xff08;只有图像的数据&#xff0c;不包…...