文档章节

Multi thread: std::promise

SHIHUAMarryMe
 SHIHUAMarryMe
发布于 2016/05/14 16:59
字数 1074
阅读 89
收藏 1

前面我们了解到可以使用std::shared_future/std::shared_future在多个其他线程中处理另外一个线程的函数的返回结果.

 

那么疑问来了,我们要怎样在多线程之间传递数据呢?

demo1 :std::promise在线程之间传递参数/处理异常:

#include <iostream>
#include <thread>
#include <chrono>
#include <future>
#include <utility>
#include <exception>
#include <stdexcept>
#include <string>

void doSomething(std::promise<std::string>&  p)
{
	try {
		std::cout << "read char('x' for exception): ";
		char c = std::cin.get();
		if (c == 'x') {
			throw  std::runtime_error(std::string("runtime_error"));
		}

		std::string str = std::string("char ") + c + std::string(" read");
		p.set_value(std::move(str));

		p.set_value(std::string("shihuawoaini"));

	}catch (...) {
		p.set_exception(std::current_exception());

	}
}

int main()
{
	try {
		std::promise<std::string> p;
		std::thread t1(doSomething, std::ref(p)); //注意这里是以引用的形式传递的std::promise对象.
        //只有以引用的形式我们才能改变std::promise中的shared state的状态,而且本事std::promise是不支持拷贝的
		t1.detach();

		std::future<std::string> f(p.get_future()); //此时的std::future和前面的std::promise共用同一个shared state;

		std::cout << "result: " << f.get() << std::endl; //注意这里的get(),当std::future的shared state未被设置为ready的时候会block mian线程.直到被设置为ready.

	}catch (const std::exception& e) {
		std::cerr << "EXCEPTION: " << e.what() << std::endl;

	}catch (...) {
		std::cerr << "EXCEPTION: " << std::endl;
	}
}

 最好自己跑一下上面的例子再看下面的内容

 

 

在向线程传递std::promise对象作为参数的时候尽量用std::ref.

1,因为std::promise对象是不支持拷贝的.

2,只有以by-reference的形式传递我们才能改变原std::promise对象的shared state.

std::promise的构造函数:

1,构造函数在构建std::promise的时候构建一个新的shared state.

2,移动构造函数从另外的std::promise对象获得shared state,被移动的std::promise对象不再具有shared state.

std::promise内的shared state存放一个相应类型的值或者异常.

promise(); //默认构造函数.

template <class Alloc> promise (allocator_arg_t aa, const Alloc& alloc);//自己设置内存分配器的构造函数.
	
promise (const promise&) = delete; //拷贝构造函数被删除掉了. 

promise (promise&& x) noexcept;//移动构造函数.

std::promise::get_future

该函数返回一个std::future对象,且std::future对象与当前std::promise共用一个shared state:

1, 也就是说当当前std::promise对象的shared state被设为ready的时候,被返回出来的std::future对象的shared state也被设置为ready.

2,也就是说我们可以通过被返回的std::future对象获得std::promise中的值和异常.

future<T> get_future();

std::promise::operator=

虽然是operator=但是只能使用移动赋值运算符.抛弃当前std::promise内的shared state,获得rhs的shared state,且被移动的std::promise对象变成不能使用状态,否则抛出std::future_error异常.

promise& operator= (promise&& rhs) noexcept;
	
promise& operator= (const promise&) = delete; //拷贝复制运算符被删除了.

std::promise::set_exception

exception_ptr是一个智能指针,存储一个指向异常的智能指针到std::promise的shared state,

且当前shared state的状态会立即被设置为ready.

void set_exception (exception_ptr p);

std::promise::set_value

设置一个值到存储到当前std::promise的shared state,且立即将shared state的状态设置为ready.

void set_value (const T& val);

void set_value (T&& val);

void promise<R&>::set_value (R& val);   // when T is a reference type (R&)

void promise<void>::set_value (void);   // when T is void

std::promise::set_exception_at_thread_exit

设置一个存储异常的智能指针存储到当前std::promise对象的shared state里,但是并没有立即把该shared state设置为ready,而是直到当前线程完成退出的时才被设置为ready.

void set_exception_at_thread_exit (exception_ptr p);

std::promise::set_value_at_thread_exit

设置一个值存储到当前std::promise对象的shared state里,但是并没有立即把该shared sate设置为ready,而是直到当前线程结束退出的时候才被设置为ready.

void set_value_at_thread_exit (const T& val);

void set_value_at_thread_exit (T&& val);

void promise<R&>::set_value_at_thread_exit (R& val);     // when T is a reference type (R&)

void promise<void>::set_value_at_thread_exit (void);     // when T is void

总结:

如果我们通过当前std::promise对象的get_future()得到一个std::future,由于是得到的std::future和当前std::promise对象是共用一个shared state的,因此当我们想要通过std::future对象的get()获得存储于shared state状态中的值的时候,如果这个shared state没有被设置为ready,就会阻塞正在调用该std::future对象get()函数的线程.

© 著作权归作者所有

共有 人打赏支持
SHIHUAMarryMe
粉丝 13
博文 164
码字总数 138212
作品 0
武汉
程序员
私信 提问
c++11 线程:让你的多线程任务更轻松

介绍 本文旨在帮助有经验的Win32程序员来了解c++ 11线程库及同步对象 和 Win32线程及同步对象之间的区别和相似之处。 在Win32中,所有的同步对象句柄(HANDLE)是全局句柄.它们可以被共享,甚...

小熊猫大暴走
2013/02/06
51K
14
C++11实现跨平台线程池

生产者消费者 转自:http://www.cnblogs.com/sanjin/archive/2012/08/09/2629890.html std::conditionvariable的使用方法如下: ·当持有锁(mutex)之后,线程调用wait ·wait解开持有的互斥锁...

初雪之音
2016/03/19
376
0
Multi thread: std::shared_future(2)

在之前我们了解了class std::future,但是通过class std::future获得的结果只能get()一次,第二次调用就会产生不可预期的结果,通常会抛出一个std::future_error。 但是当多个其他线程想处理另...

SHIHUAMarryMe
2016/05/13
96
0
Multi thread: std::async()和std::future(1)

对于初学者而言,“以多线程运行程序”的最佳起点就是C++标准库中的 std::async() 和 class std::future提供的高级接口. (1),std::async()提供一个接口,让一段机能或者说一个callable obje...

SHIHUAMarryMe
2016/05/10
292
0
C++11: std::thread和std::this_thread

C++11起,标准库除了提供std::async(),这样的高级线程接口以外,还提供了更加简单的std::thread,我们只需要声明一个std::thread对象,然后传递进去函数(function注意这里并不是callable obje...

SHIHUAMarryMe
2016/03/07
158
0

没有更多内容

加载失败,请刷新页面

加载更多

oh-my-zsh 自定义

GitHub 地址 基于 oh-my-zsh 的自定义配置,增加了一些个人常用插件与皮肤。 采用的是 git submodule 来维护,包括 oh-my-zsh,之所以这么搞,主要是手头有多台 linux 需要维护, 每台机器、...

郁也风
今天
4
0
Docker安装踩坑:E_FAIL 0x80004005的解决

参考 菜鸟教程--Windows Docker 安装 http://www.runoob.com/docker/windows-docker-install.html 官方文档-Install Docker Toolbox on Windows https://docs.docker.com/toolbox/toolbox_in......

karma123
今天
4
0
js垃圾回收机制和引起内存泄漏的操作

JS的垃圾回收机制了解吗? Js具有自动垃圾回收机制。垃圾收集器会按照固定的时间间隔周期性的执行。 JS中最常见的垃圾回收方式是标记清除。 工作原理:是当变量进入环境时,将这个变量标记为“...

Jack088
昨天
17
0
大数据教程(10.1)倒排索引建立

前面博主介绍了sql中join功能的大数据实现,本节将继续为小伙伴们分享倒排索引的建立。 一、需求 在很多项目中,我们需要对我们的文档建立索引(如:论坛帖子);我们需要记录某个词在各个文...

em_aaron
昨天
27
0
"errcode": 41001, "errmsg": "access_token missing hint: [w.ILza05728877!]"

Postman获取微信小程序码的时候报错, errcode: 41001, errmsg: access_token missing hint 查看小程序开发api指南,原来access_token是直接当作parameter的(写在url之后),scene参数一定要...

两广总督bogang
昨天
31
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部