[C++11]std::promise介绍及使用

一、std::promise介绍

std::promise C++11并发编程中常用的一个类,常配合std::future使用。其作用是在一个线程t1中保存一个类型typename T的值,可供相绑定的std::future对象在另一线程t2中获取。

二、代码示例:
下面我们通过几个简单小例子逐渐深入了解std::promise的使用方法。
示例1:
#include <iostream>
#include <future>
#include <chrono>

void Thread_Fun1(std::promise<int> &p)
{
	//为了突出效果,可以使线程休眠5s
	std::this_thread::sleep_for(std::chrono::seconds(5));

	int iVal = 233;
	std::cout << "传入数据(int):" << iVal << std::endl;

	//传入数据iVal
	p.set_value(iVal);
}

void Thread_Fun2(std::future<int> &f)
{
	//阻塞函数,直到收到相关联的std::promise对象传入的数据
	auto iVal = f.get();		//iVal = 233

	std::cout << "收到数据(int):" << iVal << std::endl;
}

int main()
{
	//声明一个std::promise对象pr1,其保存的值类型为int
	std::promise<int> pr1;
	//声明一个std::future对象fu1,并通过std::promise的get_future()函数与pr1绑定
	std::future<int> fu1 = pr1.get_future();

	//创建一个线程t1,将函数Thread_Fun1及对象pr1放在线程里面执行
	std::thread t1(Thread_Fun1, std::ref(pr1));
	//创建一个线程t2,将函数Thread_Fun2及对象fu1放在线程里面执行
	std::thread t2(Thread_Fun2, std::ref(fu1));

	//阻塞至线程结束
	t1.join();
	t2.join();

	return 1;
}

可以看到std::future对象fu1先是通过std::promise的函数get_future()与std::promise对象pr1相绑定,pr1在线程t1中通过set_value()传入共享数据,fu1在线程t2中通过阻塞函数get()获取到传入的数据。
示例1中传入的数据类型是int,前面介绍中说std::promise可以保存typename T的数据,那么可以保存函数指针吗?答案是可行的,请看示例。

示例2:
#include <iostream>
#include <future>
#include <chrono>
#include <functional>

//声明一个可调对象T
using T = std::function<int(int)>;		//等同于typedef std::function<int(int)> T;

int Test_Fun(int iVal)
{
	std::cout << "Value is:" << iVal << std::endl;
	return iVal + 232;
}

void Thread_Fun1(std::promise<T> &p)
{
	//为了突出效果,可以使线程休眠5s
	std::this_thread::sleep_for(std::chrono::seconds(5));

	std::cout << "传入函数Test_Fun" << std::endl;

	//传入函数Test_Fun
	p.set_value(std::bind(&Test_Fun, std::placeholders::_1));
}

void Thread_Fun2(std::future<T> &f)
{
	//阻塞函数,直到收到相关联的std::promise对象传入的数据
	auto fun = f.get();		//iVal = 233

	int iVal = fun(1);

	std::cout << "收到函数并运行,结果:" << iVal << std::endl;
}

int main()
{
	//声明一个std::promise对象pr1,其保存的值类型为int
	std::promise<T> pr1;
	//声明一个std::future对象fu1,并通过std::promise的get_future()函数与pr1绑定
	std::future<T> fu1 = pr1.get_future();

	//创建一个线程t1,将函数Thread_Fun1及对象pr1放在线程里面执行
	std::thread t1(Thread_Fun1, std::ref(pr1));
	//创建一个线程t2,将函数Thread_Fun2及对象fu1放在线程里面执行
	std::thread t2(Thread_Fun2, std::ref(fu1));

	//阻塞至线程结束
	t1.join();
	t2.join();

	return 1;
}

既然可以传函数对象,那么是否可以通过模板魔改,传入可变元函数?请看示例。

示例3:
#include <iostream>
#include <future>
#include <chrono>
#include <functional>

//声明一个可调对象F
using F = std::function<int(int, int, int&)>;		//等同于typedef std::function<int(int, int, int&)> F;

//函数可以改成任意参数,任意返回类型
int Test_Fun(int a, int b, int &c)
{
	//a = 1, b = 2
	c = a + b + 230;
	return c;
}

void Thread_Fun1(std::promise<F> &p)
{
	//为了突出效果,可以使线程休眠5s
	std::this_thread::sleep_for(std::chrono::seconds(5));

	std::cout << "传入函数Test_Fun" << std::endl;

	//传入函数Test_Fun
	p.set_value(std::bind(&Test_Fun, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3));
}

template<typename T, typename ...Args>
void Thread_Fun2(std::future<T> &f, Args&& ...args)
{
	//阻塞函数,直到收到相关联的std::promise对象传入的数据
	auto fun = f.get();		//fun等同于Test_Fun

	auto fResult = fun(std::forward<Args>(args)...);

	std::cout << "收到函数并运行,结果:" << fResult << std::endl;
}

int main()
{
	//声明一个std::promise对象pr1,其保存的值类型为int
	std::promise<F> pr1;
	//声明一个std::future对象fu1,并通过std::promise的get_future()函数与pr1绑定
	std::future<F> fu1 = pr1.get_future();

	//声明一个变量
	int iVal = 0;

	//创建一个线程t1,将函数Thread_Fun1及对象pr1放在线程里面执行
	std::thread t1(Thread_Fun1, std::ref(pr1));
	//创建一个线程t2,将函数Thread_Fun2及对象fu1放在线程里面执行
	std::thread t2(Thread_Fun2<F, int, int, int&>, std::ref(fu1), 1, 2, std::ref(iVal));

	//阻塞至线程结束
	t1.join();
	t2.join();

	//此时iVal的值变成233

	return 1;
}

看完代码示例,是不是感觉std::promise很有趣,C++11很有趣?2333333





  • 59
    点赞
  • 170
    收藏
    觉得还不错? 一键收藏
  • 10
    评论
### 回答1: 在C++std::promise是一种线程间通信机制,它允许一个线程在另一个线程设置一个值。在使用std::promise时,一个线程可以将一个值传递给另一个线程,而无需等待该线程返回。具体而言,std::promise允许线程A创建一个std::future对象并将其返回给线程B,线程B可以在某个时间点通过调用std::promise的set_value()方法来设置std::future对象的值。一旦std::future对象被设置了值,线程A可以通过调用std::future对象的get()方法来获取该值。在使用std::promise时,需要注意的是,一个std::promise对象只能被设置一次值,因此在使用它时需要小心。 ### 回答2: std::promiseC++标准库的一个类,用于在线程之间传递异步操作的结果。它通常与 std::future 一起使用,通过 std::promise 传递数据给 std::future 对象,在另一个线程可以获取到该数据。 std::promise 的用法可以分为三个步骤:创建 promise 对象、获取 future 对象、设置 promise 对象的值。 首先,我们需要创建一个 promise 对象,这可以通过构造函数来完成,例如:std::promise<int> myPromise; 这样就创建了一个用于传递 int 类型数据的 promise 对象。 接下来,我们需要使用 promise 对象获取对应的 future 对象,通过调用 promise 对象的成员函数 std::promise::get_future() 来获取,例如:std::future<int> myFuture = myPromise.get_future(); 这样就获得了一个用于接收 int 类型数据的 future 对象。 最后,我们可以通过设置 promise 对象的值来传递数据给 future 对象。这可以通过调用 promise 对象的成员函数 std::promise::set_value() 来完成,例如:myPromise.set_value(42); 这样就将值 42 传递给了之前创建的 future 对象。 在另一个线程,我们可以通过调用 future 对象的成员函数 std::future::get() 来获取到设置的值,例如:int result = myFuture.get(); 这样就可以获取到之前设置的值 42。 需要注意的是,一旦 promise 对象被设置了值,就不能再次设置。如果我们不希望在某些情况下传递数据到 future 对象,可以使用 std::promise::set_exception() 来设置一个异常。 总而言之,std::promise 提供了一种方便的机制来在线程之间传递异步操作的结果,通过设置 promise 对象的值,可以在另一个线程获取到这些值,并进行相应的处理。 ### 回答3: std::promiseC++标准库的一个类模板,用于在多线程编程用来存放一个值或者一个异常的容器。它通常与std::future一起使用,将一个值或异常从一个线程传递到另一个线程。 std::promise提供了两个主要的成员函数set_value()和set_exception()来设置值和异常。当一个线程调用set_value()或set_exception()时,std::promise的状态会变为“就绪”状态。其他线程可以通过std::future对象获取std::promise的值或异常。 使用std::promise的基本步骤如下: 1. 创建一个std::promise对象,例如std::promise<int> p; 2. 创建一个std::future对象,通过std::promise的get_future()成员函数获取,例如std::future<int> f = p.get_future(); 3. 启动一个线程,该线程可以在某个时间点调用p.set_value()或p.set_exception()来设置值或异常。 4. 在需要的地方,通过std::future对象的get()成员函数获取std::promise的值或异常。 例如,通过std::promise实现一个计算线程和一个读取结果的线程的示例: ``` #include <iostream> #include <thread> #include <functional> #include <future> void calculate(std::promise<int>& resultPromise, int a, int b) { int result = a + b; resultPromise.set_value(result); } int main() { std::promise<int> resultPromise; std::future<int> resultFuture = resultPromise.get_future(); std::thread calculationThread(calculate, std::ref(resultPromise), 3, 4); // 等待计算线程完成 calculationThread.join(); // 读取结果 int result = resultFuture.get(); std::cout << "Result: " << result << std::endl; return 0; } ``` 这个示例,计算线程通过调用resultPromise.set_value()来设置结果值,读取结果的线程通过resultFuture.get()来阻塞等待结果值的返回,这样就实现了两个线程之间的值的传递和同步。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值