介绍
packaged_task
类模板也是定义于future头文件中,它包装任何可调用 (Callable) 目标,包括函数、 lambda 表达式、 bind 表达式或其他函数对象,使得能异步调用它,其返回值或所抛异常被存储于能通过 std::future 对象访问的共享状态中。简言之,将一个普通的可调用函数对象转换为异步执行的任务。模板声明如下:
template< class R, class ...Args >
class packaged_task< fn(Args...)>;
其中:
- fn 是可以调用目标
- Args 函数入参
通过packaged_task
包装后,可以通过thread启动或者仿函数形式启动,其执行结果返回值或所抛异常被存储于能通过 std::future
对象访问的共享状态中。
实例
在packaged_task
成员中,进行了()运算符重载,因此我们可以直接调用这个开始启动任务。
void operator()( ArgTypes... args );
以 args 为参数调用存储的任务。
实例代码如下:
#include <iostream> // std::cout
#include <thread> // std::thread
#include <chrono>
#include <future>
using namespace std;
//普通函数
int Add(int x, int y)
{
return x + y;
}
void task_lambda()
{
//包装可调用目标时lambda
packaged_task<int(int,int)> task([](int a, int b){ return a + b;});
//仿函数形式,启动任务
task(2, 10);
//获取共享状态中的值,直到ready才能返回结果或者异常
future<int> result = task.get_future();
cout << "task_lambda :" << result.get() << "\n";
}
void task_thread()
{
//包装普通函数
std::packaged_task<int (int,int)> task(Add);
future<int> result = task.get_future();
//启动任务,非异步
task(4,8);
cout << "task_thread :" << result.get() << "\n";
//重置共享状态
task.reset();
result = task.get_future();
//通过线程启动任务,异步启动
thread td(move(task), 2, 10);
td.join();
//获取执行结果
cout << "task_thread :" << result.get() << "\n";
}
int main(int argc, char *argv[])
{
task_lambda();
task_thread();
return 0;
}
运行结果:
task_lambda :12
task_thread :12
task_thread :12
区别
那么std::async
、std::promise
、std::packaged_task
以及std::future
之间有什么联系和区别呢?
通过 std::async
和 std::promise
以及以上内容可以看出:
std::future
是用于获取将来共享状态的运行结果或异常,相当于一个中间件,std::async 、std::promise、std::packaged_task
都离不开它的帮助;
std::packaged_task
用于包装可调用目标,以便异步执行任务;
std::promise
用于设置共享状态的值,可以用于线程间交流,这个是比较特殊的。
std::async
是最优雅地方式启动任务异步执行;在多数情况下,建议使用asyn开启异步任务,而不是使用packaged_task
方式。