C++ 20 并发编程 std::promise

C++ 20 并发编程 std::promise

std::promisestd::future是一对, 通过它们可以进行更加灵活的任务控制

promise通过函数set_value()传入一个值, 异常, 或者通知, 并异步的获取结果

例子:

void product(std::promise<int>&& intPromise, int a, int b)
{
	intPromise.set_value(a * b);
}


int main(int argc, char* argv[])
{
	int a = 666;
	int b = 999;
	std::promise<int> productPromise;

	std::future<int> productResult = productPromise.get_future();

	std::jthread productThread(product, std::move(productPromise), a, b);

	std::cout << std::format("product is {}\n", productResult.get());
}

image-20220819224019977

std::future:

  • promise获取值
  • 询问值是否可用
  • 等待通知
  • 创建shared_future

成员函数

(构造函数)构造 future 对象 (公开成员函数)
(析构函数)析构 future 对象 (公开成员函数)
operator=移动future对象 (公开成员函数)
share*this 转移共享状态给 shared_future 并返回它 (公开成员函数)
获取结果
get返回结果 (公开成员函数)
状态
valid检查 future 是否拥有共享状态 (公开成员函数)
wait等待结果变得可用 (公开成员函数)
wait_for等待结果,如果在指定的超时间隔后仍然无法得到结果,则返回。 (公开成员函数)
wait_until等待结果,如果在已经到达指定的时间点时仍然无法得到结果,则返回。 (公开成员函数)

std::future_status

调用后wait_for或者wait_until返回的结果

enum class future_status
{
	ready,      //成功
	timeout,    //超时
	deferred    //延迟
};

例子:

void getAnswer(std::promise<int> intPromise)
{
	std::this_thread::sleep_for(2s);
	intPromise.set_value(100);
}


int main(int argc, char* argv[])
{
	std::promise<int> answerPromise;

	auto fut = answerPromise.get_future();

	std::jthread productThread(getAnswer, std::move(answerPromise));

	std::future_status status{};

	do
	{
		status = fut.wait_for(0.5s);
		std::cout << "结果未准备完成 \n\n";
	}
	while (status != std::future_status::ready);


	std::cout << std::format("answer is {}\n ", fut.get());
}

std::shared_future

类模板 std::shared_future 提供访问异步操作结果的机制,类似 std::future ,除了允许多个线程等候同一共享状态。不同于仅可移动的 std::future (故只有一个实例能指代任何特定的异步结果),std::shared_future 可复制而且多个 shared_future 对象能指代同一共享状态。

但即使使用std::shared_future, 若我们从多个线程访问同一个对象, 就必须采取锁来保护以避免竞争

所以我们最好向每个线程传递std::shared_future对象的副本, 保证每个线程独有, 这样就不会发生数据竞争了

网图

若每个线程通过其自身的 shared_future 对象副本访问,则从多个线程访问同一共享状态是安全的。

两种方式获取std::shared_future

  • std::shared_future<int> fut = answerPromise.get_future();
  • std::future::share
std::promise<int> p;
std::future<int>  f(p.get_future());
assert(f.valid());     future对象f有效
std::shared_future<int> sf(std::move(f));
assert(!f.valid());   对象f不再有效
assert(sf.valid());    对象sf开始生效

例子:

struct Div
{
	void operator()(std::promise<int>&& intPromise, int a, int b)
	{
		intPromise.set_value(a / b);
	}
};

struct Requestor
{
	void operator ()(std::shared_future<int> shaFut)
	{
		std::lock_guard<std::mutex> coutGuard(coutMutex);


		std::cout << "threadId(" << std::this_thread::get_id() << "): ";


		std::cout << "20/10= " << shaFut.get() << std::endl;
	}
};

int main()
{
	std::cout << std::boolalpha << std::endl;


	std::promise<int> divPromise;


	std::future<int> divResult = divPromise.get_future();

	std::cout << "divResult.valid(): " << divResult.valid() << std::endl;


	Div div;
	std::jthread divThread(div, std::move(divPromise), 20, 10);

	std::cout << "divResult.valid(): " << divResult.valid() << std::endl;

	std::shared_future<int> sharedResult = divResult.share();

	std::cout << "divResult.valid(): " << divResult.valid() << "\n\n";

	Requestor req;
	std::jthread sharedThread1(req, sharedResult);
	std::jthread sharedThread2(req, sharedResult);
	std::jthread sharedThread3(req, sharedResult);
	std::jthread sharedThread4(req, sharedResult);
	std::jthread sharedThread5(req, sharedResult);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答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::promise 是C++标准库中的一个类,用于在线程之间传递异步操作的结果。它通常与 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::promise是C++标准库中的一个类模板,用于在多线程编程中用来存放一个值或者一个异常的容器。它通常与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个

红包金额最低5元

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

抵扣说明:

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

余额充值