V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
zoule
V2EX  ›  C++

std::async 的使用总结

  •  
  •   zoule · 2021-01-25 15:32:01 +08:00 · 1575 次点击
    这是一个创建于 1430 天前的主题,其中的信息可能已经有所发展或是发生改变。

    C++98标准中并没有线程库的存在,直到C++11中才终于提供了多线程的标准库,提供了管理线程、保护共享数据、线程间同步操作、原子操作等类。多线程库对应的头文件是#include <thread>,类名为std::thread

    然而线程毕竟是比较贴近系统的东西,使用起来仍然不是很方便,特别是线程同步及获取线程运行结果上就更加麻烦。我们不能简单的通过thread.join()得到结果,必须定义一个线程共享的变量来传递结果,同时还要考虑线程间的互斥问题。好在C++11中提供了一个相对简单的异步接口std::async,通过这个接口可以简单的创建线程并通过std::future中获取结果。以往都是自己去封装线程实现自己的 async,现在有线程的跨平台接口可以使用就极大的方便了 C++多线程编程。

    先看一下std::async的函数原型

    //(C++11 起) (C++17 前)
    template< class Function, class... Args>
    std::future<std::result_of_t<std::decay_t<Function>(std::decay_t<Args>...)>>
        async( Function&& f, Args&&... args );
    
    //(C++11 起) (C++17 前)
    template< class Function, class... Args >
    std::future<std::result_of_t<std::decay_t<Function>(std::decay_t<Args>...)>>
        async( std::launch policy, Function&& f, Args&&... args );  
    
    

    第一个参数是线程的创建策略,有两种策略可供选择

    • std::launch::async:在调用 async 就开始创建线程。
    • std::launch::deferred:延迟加载方式创建线程。调用 async 时不创建线程,直到调用了 future 的 get 或者 wait 时才创建线程。

    默认策略是:std::launch::async | std::launch::deferred也就是两种策略的合集,具体什么意思后面详细再说

    第二个参数是线程函数

    线程函数可接受function, lambda expression, bind expression, or another function object

    第三个参数是线程函数的参数

    不再说明

    返回值 std::future

    std::future是一个模板类,它提供了一种访问异步操作结果的机制。从字面意思上看它表示未来,这个意思就非常贴切,因为她不是立即获取结果但是可以在某个时候以同步的方式来获取结果。我们可以通过查询 future 的状态来获取异步操作的结构。future_status 有三种状态:

    • deferred:异步操作还未开始
    • ready:异步操作已经完成
    • timeout:异步操作超时,主要用于 std::future<T>.wait_for()

    示例:

    //查询 future 的状态
    std::future_status status;
    do {
        status = future.wait_for(std::chrono::seconds(1));
        if (status == std::future_status::deferred) {
            std::cout << "deferred" << std::endl;
        } else if (status == std::future_status::timeout) {
            std::cout << "timeout" << std::endl;
        } else if (status == std::future_status::ready) {
            std::cout << "ready!" << std::endl;
        }
    } while (status != std::future_status::ready); 
    
    

    std::future获取结果的方式有三种:

    • get:等待异步操作结束并返回结果
    • wait:等待异步操作结束,但没有返回值
    • waite_for:超时等待返回结果,上面示例中就是对超时等待的使用展示

    介绍完了std::async的函数原型,那么它到底该如何使用呢?

    std::async的基本用法:示例链接

    #include <iostream>
    #include <vector>
    #include <algorithm>
    #include <numeric>
    #include <future>
    #include <string>
    #include <mutex>
    
    std::mutex m;
    struct X {
        void foo(int i, const std::string& str) {
            std::lock_guard<std::mutex> lk(m);
            std::cout << str << ' ' << i << '\n';
        }
        void bar(const std::string& str) {
            std::lock_guard<std::mutex> lk(m);
            std::cout << str << '\n';
        }
        int operator()(int i) {
            std::lock_guard<std::mutex> lk(m);
            std::cout << i << '\n';
            return i + 10;
        }};
    
    template <typename RandomIt>int parallel_sum(RandomIt beg, RandomIt end){
        auto len = end - beg;
        if (len < 1000)
            return std::accumulate(beg, end, 0);
    
        RandomIt mid = beg + len/2;
        auto handle = std::async(std::launch::async,
                                 parallel_sum<RandomIt>, mid, end);
        int sum = parallel_sum(beg, mid);
        return sum + handle.get();
    }
    
    int main(){
        std::vector<int> v(10000, 1);
        std::cout << "The sum is " << parallel_sum(v.begin(), v.end()) << '\n';
    
        X x;
        // 以默认策略调用 x.foo(42, "Hello") :
        // 可能同时打印 "Hello 42" 或延迟执行
        auto a1 = std::async(&X::foo, &x, 42, "Hello");
        // 以 deferred 策略调用 x.bar("world!")
        // 调用 a2.get() 或 a2.wait() 时打印 "world!"
        auto a2 = std::async(std::launch::deferred, &X::bar, x, "world!");
        // 以 async 策略调用 X()(43) :
        // 同时打印 "43"
        auto a3 = std::async(std::launch::async, X(), 43);
        a2.wait();                     // 打印 "world!"
        std::cout << a3.get() << '\n'; // 打印 "53"
    } // 若 a1 在此点未完成,则 a1 的析构函数在此打印 "Hello 42"
    
    

    可能的结果

    The sum is 10000
    43
    world!
    53
    Hello 42
    
    

    由此可见,std::async是异步操作做了一个很好的封装,使我们不用关注线程创建内部细节,就能方便的获取异步执行状态和结果,还可以指定线程创建策略。

    深入理解线程创建策略

    • std::launch::async 调度策略意味着函数必须异步执行,即在另一线程执行。
    • std::launch::deferred 调度策略意味着函数可能只会在 std::async 返回的 future 对象调用 get 或 wait 时执行。那就是,执行会推迟到其中一个调用发生。当调用 get 或 wait 时,函数会同步执行,即调用者会阻塞直到函数运行结束。如果 get 或 wait 没有被调用,函数就绝对不会执行。

    两者策略都很明确,然而该函数的默认策略却很有趣,它不是你显示指定的,也就是第一个函数原型中所用的策略即std::launch::async | std::launch::deferred,c++标准中给出的说明是:

    进行异步执行还是惰性求值取决于实现

    auto future = std::async(func);        // 使用默认发射模式执行 func
    
    

    这种调度策略我们没有办法预知函数 func 是否会在哪个线程执行,甚至无法预知会不会被执行,因为 func 可能会被调度为推迟执行,即调用 get 或 wait 的时候执行,而 get 或 wait 是否会被执行或者在哪个线程执行都无法预知。

    同时这种调度策略的灵活性还会混淆使用 thread_local 变量,这意味着如果 func 写或读这种线程本地存储(Thread Local Storage,TLS),预知取到哪个线程的本地变量是不可能的。

    它也影响了基于 wait 循环中的超时情况,因为调度策略可能为deferred的,调用 wait_for 或者 wait_until 会返回值 std::launch::deferred 。这意味着下面的循环,看起来最终会停止,但是,实际上可能会一直运行:

    void func()           // f 睡眠 1 秒后返回
    {
        std::this_thread::sleep_for(1);
    }
    auto future = std::async(func);      // (概念上)异步执行 f
    while(fut.wait_for(100ms) !=         // 循环直到 f 执行结束
          std::future_status::ready)     // 但这可能永远不会发生
    {
        ...
    }
    
    

    为避免陷入死循环,我们必须检查 future 是否把任务推迟,然而 future 无法获知任务是否被推迟,一个好的技巧就是通过 wait_for(0)来获取 future_status 是否是 deferred:

    auto future = std::async(func);      // (概念上)异步执行 f
    if (fut.wait_for(0) == std::future_status::deferred)  // 如果任务被推迟
    {
        ...     // fut 使用 get 或 wait 来同步调用 f
    } else {            // 任务没有被推迟
        while(fut.wait_for(100ms) != std::future_status::ready) { // 不可能无限循环
          ...    // 任务没有被推迟也没有就绪,所以做一些并发的事情直到任务就绪
        }
        ...        // fut 就绪
    }
    
    

    有人可能会说既然有这么多缺点为啥还要用它,因为毕竟我们考虑的极限情况下的可能,有时候我不要求它是并发还是同步执行,也不需要考虑修改那个线程 thread_local 变量,同时也能接受可能任务永远不会执行,那么这种方式就是一种方便且高效的调度策略。

    综上所述,我们总结出以下几点:

    • std::async 的默认调度策略既允许任务异步执行,又允许任务同步执行。
    • 默认策略灵活性导致了使用 thread_local 变量时的不确定性,它隐含着任务可能不会执行,它还影响了基于超时的 wait 调用的程序逻辑。
    • 如果异步执行是必需的,指定 std::launch::async 发射策略。

    参考文章:

    API Reference Document

    用 C++11 的 std::async 代替线程的创建

    目前尚无回复
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   实用小工具   ·   3065 人在线   最高记录 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 25ms · UTC 13:33 · PVG 21:33 · LAX 05:33 · JFK 08:33
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.