C++ std::future

std::future是用来接收一个线程的执行结果的,并且是一次性的。

共享状态shared state

future可以关联一个共享状态,共享状态是用来储存要执行结果的。这个结果是async、promise、packaged_task设置的,且这个结果只能设置一次。

创建future

创建future有四种方式

  1.  直接构造函数创建,这种创建的future没用共享状态。
    std::future<DataType> future;
  2. 调用std::async,返回一个future对象,返回的future有共享状态。std::async是执行一个异步任务,这个任务会放到线程池中的线程执行,并通过future返回执行结果。通常异步任务优先考虑使用std::async,而不是std::thread。
    auto future = std::async([]{return DataType{5};
    });
    std::cout <<"future is Valid:" << future.valid() << " , async Result:" << future.get().n << std::endl;
  3. 创建std::promise,然后调用std::promise::get_future获取future对象,返回的future有共享状态。promise可以设置一个值传递给future,也可以设置一个异常传递给future。promise的set_value和set_exception 只能调用一次,调第二次时会抛出异常。如果promise在析构前都未设置结果,调用future::get获取结果会抛出异常。
        {std::cout << "------------- promise -----------------\n";std::promise<DataType> promise;promise.set_value({5});// promise.set_value({6}); //set_value只能调用一次,再次调用会抛出异常auto future = promise.get_future();// auto future2 = promise.get_future(); //get_future只能调用一次,再次调用会抛出异常std::cout <<"future is Valid:" << future.valid() << " , async Result:" << future.get().n << std::endl;}{std::cout << "------------- promise exception -----------------\n";std::promise<DataType> promise;try {throw std::runtime_error("error");} catch (...) {// 如果调用了 promise.set_value ,则不能调用 promise.set_exception ,// 否则 promise.set_exception 会抛出异常promise.set_exception(std::current_exception());}auto future = promise.get_future();try {auto data =  future.get();std::cout <<"future is Valid:" << future.valid() << " , async Result:" << data.n << std::endl;} catch (const std::exception& e) {std::cout << e.what() << std::endl;}}
    
  4. 创建std::packaged_task,然后调用get_futurestd::packaged_task::get_future获取future对象,返回的future有共享状态。
    std::cout << "------------- packaged -----------------\n";
    std::packaged_task<DataType(int)> task([](int a) {return DataType{a};
    });
    auto future = task.get_future();
    // auto future2 = task.get_future(); //get_future只能调用一次,再次调用会抛出异常
    task(5);
    std::cout <<"future is Valid:" << future.valid()<< " , async Result:" << future.get().n << std::endl;

future状态

可以通过函数future::valid可以判断future是否有共享状态。如果有共享状态可以进一步通过函数future::wait_for(0s)来判读结果是否已经设置。wait_for有三个返回值:

  1. future_status::deferred 表示结果还未设置
  2. future_status::ready 表示结果已经设置
  3. future_status::timeout 表示超时

获取结果

通过future::get函数可以获取future的结果,调用get函数需要注意以下几点:

  1.  只有有共享状态的future才能获取结果,如果没有共享状态会抛出异常。
  2. 如果获取数据时future结果已经设置好,则会立即返回。
  3. 如果future状态还未设置好,则会挂起线程一直等待,直到结果被设置。如果相应async、promise、packaged_task到结束都一直未设置状态,则会抛出异常。
  4. get函数只能调用一次,调用完成后则会删除共享状态,再次调用get函数,会因为没有共享状态而抛出异常。

也可以调用wait,wait_for, wait_until等带结果被设置。

多线程调用future

获取future结果时,可以先判断future是否有共享状态,然后再调用get函数获取结果,代码如下:

        if (future.valid()){future.get();}

上面代码在单线程调用没有问题,如果在线程同时调用则会有问题,多个线程会同时判断future有共享状态,因此同时有多个线程调用get函数,而get函数只能被调用一次,从而引发异常。

可以通过加锁解决这个问题,但这并不是好的方法。更优雅的方法是通过future::share函数,为每个线程创建一个std::shared_future。future调用完share将不再拥有共享状态,如下:

如果future本身没有共享状态,调用share会生成一个没有共享状态的shared_future。

shared_future调用get函数时并不会清楚共享状态,所以shared_future可以多次调用get函数。

示例代码:

    std::promise<DataType> promise;auto future = promise.get_future();auto shadedFuture = future.share();std::cout << "future is valid: " << future.valid() << std::endl;std::thread threads[4];for (int i = 0; i < 4; ++i){threads[i] = std::thread([future=shadedFuture](int i){std::cout << std::to_string(i) + ", future is Valid:" + std::to_string(future.valid())+ " , async Result:" + std::to_string(future.get().n) + "\n";}, i);}std::this_thread::sleep_for(std::chrono::milliseconds{100});promise.set_value({5});for (int i = 0; i < 4; ++i){threads[i].join();}std::cout << std::endl;

future析构阻塞

当future是由async创建的,且future/shared_future是最后一个指向共享状态的future,如果async还在执行,则future在析构时会阻塞当前线程,等待async执行完成。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/108067.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

NLP机器翻译全景:从基本原理到技术实战全解析

目录 一、机器翻译简介1. 什么是机器翻译 (MT)?2. 源语言和目标语言3. 翻译模型4. 上下文的重要性 二、基于规则的机器翻译 (RBMT)1. 规则的制定2. 词典和词汇选择3. 限制与挑战4. PyTorch实现 三、基于统计的机器翻译 (SMT)1. 数据驱动2. 短语对齐3. 评分和选择4. PyTorch实现…

【Linux-day11-线程的创建与同步】

Linux 线程的创建与同步 线程的概念 线程是进程内部的一条执行序列或执行路径&#xff0c;一个进程可以包含多条线程。 进程与线程的区别 进程是资源分配的最小单位&#xff0c;线程是 CPU 调度的最小单位进程有自己的独立地址空间&#xff0c;线程共享进程中的地址空间进…

数据结构基础7:二叉树【链式结构】实现和递归思想。

二叉树的链式结构实现 一.二叉树链式结构的实现&#xff1a;1.前置说明&#xff1a;1.创建二叉树&#xff1a;2.二叉树的结构&#xff1a; 2.二叉树的遍历&#xff1a;1.二叉树的前中后序遍历&#xff1a;2.内容拓展&#xff1a; 二.二叉树链式(题目)题目一&#xff1a;计算节点…

yolov7简化网络yaml配置文件

yolov7代码结构简单&#xff0c;效果还好&#xff0c;但是动辄超过70几个模块的配置文件对于想要对网络进行魔改的朋友还是不怎么友好的&#xff0c;使用最小的tiny也有77个模块 代码的整体结构简单&#xff0c;直接将ELAN结构化写成一个类就能像yolov5一样仅仅只有20几个模块&…

【操作系统】进程控制

进程控制&#xff1a;创建新进程&#xff0c;撤销已有进程&#xff0c;实现进程状态转换等。 原语&#xff1a;进程控制用的程序段。执行期间不允许中断&#xff0c;用&#xff02;关中断&#xff02;和&#xff02;开中断&#xff02;指令&#xff08;特权指令&#xff09;实…

Android性能优化之应用瘦身(APK瘦身)

关于作者&#xff1a;CSDN内容合伙人、技术专家&#xff0c; 从零开始做日活千万级APP。 专注于分享各领域原创系列文章 &#xff0c;擅长java后端、移动开发、人工智能等&#xff0c;希望大家多多支持。 目录 一、导读二、概览2.1 apk组成 三、优化方向3.1 源代码3.1.1 代码混…

Dedecms最新版--0day分享分析(二)

前言 接上一篇的Tricks&#xff0c;既然利用远程文件下载方式成为了实现RCE的最好方法&#xff0c;毕竟在执行的时候没有恶意shell文件&#xff0c;恶意木马被存放于远端服务器&#xff0c;那么下文的day就是对远程恶意文件的利用。 环境 下载最新版本&#xff1a; https://…

Kotlin Files Paths write ByteArray writeString写多行BufferedWriter

Kotlin Files Paths write ByteArray writeString写多行BufferedWriter import java.nio.file.Files import java.nio.file.Paths import java.nio.file.StandardOpenOptionfun main(args: Array<String>) {val filePath "./myfile.txt"val path Paths.get(…

ModuleNotFoundError: No module named ‘gevent‘

1、先确定pip版本&#xff1a; pip3 list: 看到没有gevent包 如果pip版本不是最新版可以使用命令python -m pip install --upgrade pip进行更新&#xff0c; 2、安装 pip3 install gevent 安装完成

Java密码学之加解密

前篇&#xff1a;Java密码学之数字签名_东皋长歌的博客-CSDN博客 日常开发中用的比较多的功能点&#xff0c;加解密数据&#xff0c;用Java实现也是很快很实用。 下面记录一下加解密数据的过程。 1&#xff0c;创建密钥对生成器 KeyPairGenerator keyPairGen KeyPairGener…

第29章_瑞萨MCU零基础入门系列教程之改进型环形缓冲区

本教程基于韦东山百问网出的 DShanMCU-RA6M5开发板 进行编写&#xff0c;需要的同学可以在这里获取&#xff1a; https://item.taobao.com/item.htm?id728461040949 配套资料获取&#xff1a;https://renesas-docs.100ask.net 瑞萨MCU零基础入门系列教程汇总&#xff1a; ht…

Qt应用开发(基础篇)——普通按钮类 QPushButton QCommandLinkButton

一、前言 QPushButton类继承于QAbstractButton&#xff0c;是一个命令按钮的小部件。 按钮基类 QAbstractButton 按钮或者命令按钮是所有图形界面框架最常见的部件&#xff0c;当按下按钮的时候触发命令、执行某些操作或者回答一个问题&#xff0c;典型的按钮有OK&#xff0c;A…