前往小程序,Get更优阅读体验!
立即前往
发布
社区首页 >专栏 >C++异步future

C++异步future

作者头像
用户11029129
发布2025-01-27 00:18:35
发布2025-01-27 00:18:35
7000
代码可运行
举报
文章被收录于专栏:编程学习编程学习
运行总次数:0
代码可运行

🌎 C++11异步futrue

🚀future介绍

  std::future是C++11标准库中的⼀个模板类,它表⽰⼀个异步操作的结果。当我们在多线程编程中使⽤异步任务时,std::future可以帮助我们在需要的时候获取任务的执⾏结果。std::future的⼀个重要特性是能够阻塞当前线程,直到异步操作完成,从⽽确保我们在获取结果时不会遇到未完成的操作。


🚀应用场景

  • 异步任务: 当我们需要在后台执⾏⼀些耗时操作时,如⽹络请求或计算密集型任务等,std::future可以⽤来表⽰这些异步任务的结果。通过将任务与主线程分离,我们可以实现任务的并⾏处理,从⽽提⾼程序的执⾏效率。
  • 并发控制: 在多线程编程中,我们可能需要等待某些任务完成后才能继续执⾏其他操作。通过使⽤std::future,我们可以实现线程之间的同步,确保任务完成后再获取结果并继续执⾏后续操作。
  • 结果获取:std::future提供了⼀种安全的⽅式来获取异步任务的结果。我们可以使⽤std::future::get()函数来获取任务的结果,此函数会阻塞当前线程,直到异步操作完成。这样,在调⽤get()函数时,我们可以确保已经获取到了所需的结果。

🚀future操作

  一个main thread可以顺序执行多个IO操作,但是执行IO操作是非常耗费时间的,而我们又恰好只是想要IO操作的结果,所以main thread就可以通过创建child thread来执行IO,再通过future来获取IO结果:

  std::future本质上不是一个异步任务,而是一个辅助我们获取异步任务结果的东西。

  std::future并不能单独使用,必须搭配一些能够执行异步任务的模版类或函数一起使用,异步任务使用搭配:

  • std::async函数模版:异步执行一个函数,返回函数对象,获取函数执行结果。
  • std::packaged_task类模版:为一个函数生成一个异步任务对象(可调用对象),用于在其他线程中执行。
  • std::promise类模版:实例化的对象可以返回一个future,在其他线程中向promise对象设置数据,其他线程的关联future就可以获取数据。

🚩std::async函数模版

  std::async是⼀种 将任务与std::future关联 的简单⽅法。它创建并运⾏⼀个异步任务,并返回⼀个与该任务结果关联的std::future对象。默认情况下,std::async是否启动⼀个新线程,或者在等待future时,任务是否同步运⾏都取决于你给的 参数。这个参数为std::launch类型:

  • std::launch::deferred:表明该函数会被延迟调⽤,直到在future上调⽤get()或者wait()才会开始执⾏任务。
  • std::launch::async: 表明函数会在⾃⼰创建的线程上运⾏。
  • std::launch::deferred | std::launch::async: 内部通过系统等条件⾃动选择策略。
代码语言:javascript
代码运行次数:0
复制
#include <iostream>
#include <future>

int Add(int num1, int num2)
{
    std::cout << "into Add()!" << std::endl;
    return num1 + num2;
}

int main()
{
    // std::launch::async策略:内部创建一个线程执行函数,函数运行结果通过future获取
    // std::launch::deferred策略:同步策略,获取结果的时候再去执行任务
    // std::future<int> res = std::async(std::launch::async, Add, 11, 22);// 进行了一个异步非阻塞调用,调用后直接执行
    std::future<int> res = std::async(std::launch::deferred, Add, 11, 22);// 进行同步调用,调用后等待wait或get才会执行
    std::this_thread::sleep_for(std::chrono::seconds(1));
    std::cout << "——————————————————————————————————" << std::endl;
    // std::future<int>::get() : 用于获取异步任务的结果,如果还没有结果就会阻塞
    std::cout << res.get() << std::endl;
    return 0;
}

🚩std::packaged_task类模版

  std::packaged_task就是将任务和 std::feature 绑定在⼀起的模板,是⼀种对任务的封装。我们可以通过std::packaged_task对象获取任务相关联的std::feature对象,通过调⽤ get_future() 方法获得。std::packaged_task的模板参数是函数签名。可以把std::future和std::async看成是分开的,⽽ std::packaged_task则是⼀个整体。

std::async是一个模版函数,内部会创建线程执行异步任务,而std::packaged_task是一个模版类,一个任务包,是对一个函数进行二次封装,封装成为一个可调用对象作为任务放到其他线程执行的。任务包封装好了以后,可以在任意位置进行调用,通过关联的future来获取执行结果

代码语言:javascript
代码运行次数:0
复制
#include <iostream>
#include <future>
#include <thread>

int Add(int num1, int num2)
{
    std::cout << "into Add()!" << std::endl;
    return num1 + num2;
}

int main()
{
    // 1. 封装任务
    std::packaged_task<int(int, int)> task(Add);// 封装成task任务包

    // 2. 获取任务包关联的future对象
    std::future<int> res = task.get_future();

    // 2. 执行任务
    task(11, 22);

    // 3. 获取结果
    std::cout << res.get() << std::endl;

    return 0;
}

  这是在main thread中执行task,但是我们想要的是可以异步执行任务,所以创建一个线程来进行异步执行任务。

  尽量不要把任务函数当成线程的入口函数,这样每次执行任务创建线程,任务结束线程也会销毁,如果任务过于调用频繁会导致线程不断的创建销毁。比较好的方式是把任务放在线程池当中让去不断的执行任务。

  创建线程,以匿名函数作为线程的入口函数,内部再调用task任务包,单但是lambda表达式不能直接传task(调用拷贝构造)。   因为std::packaged_task不允许拷贝构造,所以我们可以通过传递指针的方式防止拷贝构造发生。同时task如果在不同作用域,需要考虑作用域的问题(res获取不到get_future),不能直接传裸的指针,可以通过智能指针进行管理并传参。

代码语言:javascript
代码运行次数:0
复制
#include <iostream>
#include <future>
#include <thread>
#include <memory>

int Add(int num1, int num2)
{
    std::cout << "into Add()!" << std::endl;
    return num1 + num2;
}

int main()
{
    // 1. 封装任务
    auto task = std::make_shared<std::packaged_task<int(int, int)>>(Add);// 封装成task任务包, 创建智能指针进行管理

    // 2. 获取任务包关联的future对象
    std::future<int> res = task->get_future();

    std::thread trd([task](){ // 匿名函数作为线程入口函数,让线程来执行封装后的任务
        (*task)(11, 22);
    });

    // 3. 获取结果
    std::cout << res.get() << std::endl;
    trd.join();
    return 0;
}

🚩std::promise类模版

  std::promise提供了⼀种设置值的⽅式,它可以在设置之后通过相关联的std::future对象进⾏读取。换种说法就是之前说过std::future可以读取⼀个异步函数的返回值, 但是要等待就绪, ⽽std::promise就提供⼀种方式⼿动让 std::future就绪。

std::promise是一个模版类,是对于结果的封装:

  1. 在使用的时候,先实例化一个指定结果的primise对象。
  2. 通过promise对象,获取关联的future对象。
  3. 在任意位置给promise设置数据,就可以通过关联的future获取到这个设置的数据。
代码语言:javascript
代码运行次数:0
复制
#include <iostream>
#include <thread>
#include <future>
#include <memory>

int Add(int num1, int num2)
{
    std::cout << "into Add()!" << std::endl;
    return num1 + num2;
}

int main()
{
    // 1. 在使用的时候,先实例化一个指定结果的primise对象。
    std::promise<int> pro;
    // 2. 通过promise对象,获取关联的future对象。
    std::future<int> ret = pro.get_future();
    // 3. 在任意位置给promise设置数据,就可以通过关联的future获取到这个设置的数据。
    std::thread td([&pro](){
        int sum = Add(11, 22);
        pro.set_value(sum);
    });

    std::cout << ret.get() << std::endl;
    td.join();
    return 0;
}

  同理,依旧需要对不同作用域进行考虑,所以使用智能指针还是比较安全的,这里就不再赘述。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2025-01-25,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 🌎 C++11异步futrue
    • 🚀future介绍
    • 🚀应用场景
    • 🚀future操作
      • 🚩std::async函数模版
      • 🚩std::packaged_task类模版
      • 🚩std::promise类模版
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档