c++ 11 promise

目录

            一 promise
            二 定义
            三 成员函数
            四 get_future
            五 设置结果
                1. set_value
                2. set_value_at_thread_exit
                3. set_exception
                4. set_exception_at_thread_exit
            六 参考

一 promise

前文 C++11 async 可知,异步操作的方式之一,是通过 std::async 接口调用可执行对象,然后通过 std::future 获得结果。
另一种常见方式是在指定的 std::thread 中执行,如何获取在 std::thread 中的执行结果?办法之一就是通过 std::promise。

二 定义

// 头文件
template< class R > class promise; (1)(C++11 起) // 空模板
template< class R > class promise; (2)(C++11 起) // 非 void 特化,用于在线程间交流对象
template<> class promise; (3)(C++11 起) // void 特化,用于交流无状态事件

1
2
3
4

类模板 std::promise 提供存储值或异常的设施。

promise 对象与其 get_future() 接口返回的 future 对象构成 promise-future 交流通道。通道的 promise对象与 future 对象关联同一共享状态(shared state), promise 对象是通道的 push 端,future 对象是通道的 pop 端。promise 存储值或异常到共享状态中,使共享状态就绪,future 通过get()异步获得结果。

promise 对共享状态的三种处理

    使就绪: promise 存储结果或异常于共享状态。标记共享状态为就绪,并解除阻塞任何等待于与该共享状态关联的 future 上的线程。
    释放: promise 放弃其对共享状态的引用。若这是最后一个这种引用,则销毁共享状态。除非这是 std::async 所创建的未就绪的共享状态,否则此操作不阻塞。
    抛弃: promise 存储以 std::future_errc::broken_promise 为 error_code 的 std::future_error 类型异常,令共享状态为就绪,然后释放它。

std::promise 只应当使用一次。

    Note that the std::promise object is meant to be used only once.

三 成员函数

构造

promise(); (1)(C++11 起)
template< class Alloc >
promise( std::allocator_arg_t, const Alloc& alloc ); (2)(C++11 起)
promise( promise&& other ) noexcept; (3)(C++11 起)
promise( const promise& other ) = delete; (4)(C++11 起) // 不可复制构造
    1
    2
    3
    4
    5

析构

~promise(); (C++11 起) // 抛弃共享状态
    1

若共享状态就绪,则释放它。若共享状态未就绪,则抛弃。

operator=

promise& operator=( promise&& other ) noexcept; (1)(C++11 起)
promise& operator=( const promise& rhs ) = delete; (2)(C++11 起) // 不可复制赋值
    1
    2

四 get_future

std::future get_future(); (C++11 起)

1

若无共享状态,或已调用 get_future 则抛出异常。即仅可调用一次。
可通过 get_future 间接获得结果。

五 设置结果

  1. set_value 设置结果为指定值, 并使状态就绪。

// (仅为泛型 promise 模板的成员)
void set_value( const R& value ) (1)(C++11 起)
void set_value( R&& value ); (2)(C++11 起)

// (仅为 promise 模板特化的成员)
void set_value( R& value ); (3)(C++11 起)
// (仅为 promise 模板特化的成员)
void set_value(); (4)(C++11 起)

1
2
3
4
5
6
7
8

demo

include // sort

include // future promise

include // cout cin endl

include // istream_iterator back_inserter

include // istringstream

include // thread

include // vector

include // accumulate

void accumulate(std::vector::iterator first,
std::vector::iterator last,
std::promise accumulate_promise) {
int sum = std::accumulate(first, last, 0);
accumulate_promise.set_value(sum); // 原子地存储 num 到共享状态,并令状态就绪
}
int main() {
std::istringstream iss_numbers{“3 4 1 42 23 -23 93 2 -289 93”};
std::vector numbers;
{
// set_value (4)
std::promise numbers_promise;
std::future numbers_future = numbers_promise.get_future();

std::thread t([&] {
  std::copy(std::istream_iterator<int>{iss_numbers},
            std::istream_iterator<int>{}, std::back_inserter(numbers));
  numbers_promise.set_value(); // 使状态就绪
});

numbers_future.wait();
std::sort(numbers.begin(), numbers.end());

for (int num : numbers)
  std::cout << num << ' ';
std::cout << std::endl;

t.join();

}
{
// set_value (1)
std::promise accumulate_promise;
std::future accumulate_future = accumulate_promise.get_future();
std::thread t(accumulate, numbers.begin(), numbers.end(),
std::move(accumulate_promise));

std::cout << "result=" << accumulate_future.get() << '\n';
t.join(); 

}
}

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49

结果

-289 -23 1 2 3 4 23 42 93 93
result=-51

1
2
  1. set_value_at_thread_exit 原子地存储 value 到共享状态,而不立即令状态就绪。在当前线程退出时,销毁所有拥有线程局域存储期的对象后,再令状态就绪。

// (仅为泛型 promise 模板的成员)
void set_value_at_thread_exit( const R& value );(1) (C++11 起)
void set_value_at_thread_exit( R&& value );(2) (C++11 起)

// (仅为 promise 模板特化的成员)
void set_value_at_thread_exit( R& value );(3) (C++11 起)
// (仅为 promise 模板特化的成员)
void set_value_at_thread_exit();(4) (C++11 起)

1
2
3
4
5
6
7
8

demo

// set_value_at_thread_exit
using namespace std::chrono_literals;
std::promise p;
std::future f = p.get_future();
std::thread([&p] {
std::this_thread::sleep_for(1s);
p.set_value_at_thread_exit(9);
}).detach();

std::cout << “Waiting…” << std::flush;
f.wait();
std::cout << “Done!\nResult is: ” << f.get() << std::endl;

1
2
3
4
5
6
7
8
9
10
11
12

结果

Waiting…Done!
Result is: 9

1
2
  1. set_exception 存储异常指针 p 到共享状态中,并令状态就绪。

void set_exception( std::exception_ptr p );(C++11 起)

1

demo

// set_exception
std::promise p;
std::future f = p.get_future();

std::thread t([&p] {
try {
throw std::runtime_error(“Example”);
} catch (…) {
try {
// store anything thrown in the promise
p.set_exception(std::current_exception());
} catch (…) {
} // set_exception() may throw too
}
});

try {
std::cout << f.get();
} catch (const std::exception& e) {
std::cout << “Exception from the thread: ” << e.what() << std::endl;
}
t.join();

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22

结果

Exception from the thread: Example

1
  1. set_exception_at_thread_exit 存储异常指针 p 到共享状态中,而不立即使状态就绪。在当前线程退出时,销毁所有拥有线程局域存储期的变量后,再零状态就绪。

六 参考

cppreference-promise