-
Notifications
You must be signed in to change notification settings - Fork 75
/
bank_account_7.cpp
117 lines (107 loc) · 3.15 KB
/
bank_account_7.cpp
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
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
#include <experimental/await>
#include <experimental/future>
#include <experimental/strand>
#include <iostream>
#include <vector>
using std::experimental::await_context;
using std::experimental::dispatch;
using std::experimental::strand;
using std::experimental::system_executor;
using std::experimental::use_future;
using std::experimental::wrap;
// Active object sharing a system-wide pool of threads.
// The caller chooses how to wait for the operation to finish.
// Lightweight, immediate execution using dispatch.
// Composition using resumable functions / stackless coroutines.
class bank_account
{
int balance_ = 0;
mutable strand<system_executor> ex_;
public:
template <class CompletionToken>
auto deposit(int amount, CompletionToken&& token)
{
return dispatch(ex_, [=]
{
balance_ += amount;
},
std::forward<CompletionToken>(token));
}
template <class CompletionToken>
auto withdraw(int amount, CompletionToken&& token)
{
return dispatch(ex_, [=]
{
if (balance_ >= amount)
balance_ -= amount;
},
std::forward<CompletionToken>(token));
}
template <class CompletionToken>
auto balance(CompletionToken&& token) const
{
return dispatch(ex_, [=]
{
return balance_;
},
std::forward<CompletionToken>(token));
}
template <class CompletionToken>
auto transfer(int amount, bank_account& to_acct, CompletionToken&& token)
{
return dispatch(
wrap(ex_, [=]
{
if (balance_ >= amount)
{
balance_ -= amount;
return amount;
}
return 0;
}),
wrap(to_acct.ex_,
[&to_acct](int deducted)
{
to_acct.balance_ += deducted;
}),
std::forward<CompletionToken>(token));
}
};
template <class Iterator, class CompletionToken>
auto find_largest_account(Iterator begin, Iterator end, CompletionToken&& token)
{
return dispatch(
[i = begin, end, largest_acct = end, balance = int(), largest_balance = int()]
(await_context ctx) mutable
{
reenter (ctx)
{
for (; i != end; ++i)
{
await balance = i->balance(ctx);
if (largest_acct == end || balance > largest_balance)
{
largest_acct = i;
largest_balance = balance;
}
}
}
return largest_acct;
},
std::forward<CompletionToken>(token));
}
int main()
{
std::vector<bank_account> accts(3);
accts[0].deposit(20, use_future).get();
accts[1].deposit(30, use_future).get();
accts[2].deposit(40, use_future).get();
accts[0].withdraw(10, use_future).get();
accts[1].transfer(5, accts[0], use_future).get();
accts[2].transfer(15, accts[1], use_future).get();
std::cout << "Account 0 balance = " << accts[0].balance(use_future).get() << "\n";
std::cout << "Account 1 balance = " << accts[1].balance(use_future).get() << "\n";
std::cout << "Account 2 balance = " << accts[2].balance(use_future).get() << "\n";
auto largest = find_largest_account(accts.begin(), accts.end(), use_future).get();
std::cout << "Largest balance = " << largest->balance(use_future).get() << "\n";
}