CP.mess
CP.mess: Message passing
The standard-library facilities are quite low-level, focused on the needs of close-to the hardware critical programming using thread
s, mutex
es, atomic
types, etc.
Most people shouldn't work at this level: it's error-prone and development is slow.
If possible, use a higher level facility: messaging libraries, parallel algorithms, and vectorization.
This section looks at passing messages so that a programmer doesn't have to do explicit synchronization.
Message passing rules summary:
- CP.60: Use a
future
to return a value from a concurrent task - CP.61: Use
async()
to spawn concurrent tasks - message queues
- messaging libraries
???? should there be a "use X rather than std::async
" where X is something that would use a better specified thread pool?
??? Is std::async
worth using in light of future (and even existing, as libraries) parallelism facilities? What should the guidelines recommend if someone wants to parallelize, e.g., std::accumulate
(with the additional precondition of commutativity), or merge sort?
CP.60: Use a future
to return a value from a concurrent task
Reason
A future
preserves the usual function call return semantics for asynchronous tasks.
There is no explicit locking and both correct (value) return and error (exception) return are handled simply.
Example
???
Note
???
Enforcement
???
CP.61: Use async()
to spawn concurrent tasks
Reason
Similar to R.12, which tells you to avoid raw owning pointers, you should
also avoid raw threads and raw promises where possible. Use a factory function such as std::async
,
which handles spawning or reusing a thread without exposing raw threads to your own code.
Example
int read_value(const std::string& filename)
{
std::ifstream in(filename);
in.exceptions(std::ifstream::failbit);
int value;
in >> value;
return value;
}
void async_example()
{
try {
std::future<int> f1 = std::async(read_value, "v1.txt");
std::future<int> f2 = std::async(read_value, "v2.txt");
std::cout << f1.get() + f2.get() << '\n';
} catch (const std::ios_base::failure& fail) {
// handle exception here
}
}
Note
Unfortunately, std::async
is not perfect. For example, it doesn't use a thread pool,
which means that it might fail due to resource exhaustion, rather than queuing up your tasks
to be executed later. However, even if you cannot use std::async
, you should prefer to
write your own future
-returning factory function, rather than using raw promises.
Example (bad)
This example shows two different ways to succeed at using std::future
, but to fail
at avoiding raw std::thread
management.
void async_example()
{
std::promise<int> p1;
std::future<int> f1 = p1.get_future();
std::thread t1([p1 = std::move(p1)]() mutable {
p1.set_value(read_value("v1.txt"));
});
t1.detach(); // evil
std::packaged_task<int()> pt2(read_value, "v2.txt");
std::future<int> f2 = pt2.get_future();
std::thread(std::move(pt2)).detach();
std::cout << f1.get() + f2.get() << '\n';
}
Example (good)
This example shows one way you could follow the general pattern set by
std::async
, in a context where std::async
itself was unacceptable for
use in production.
void async_example(WorkQueue& wq)
{
std::future<int> f1 = wq.enqueue([]() {
return read_value("v1.txt");
});
std::future<int> f2 = wq.enqueue([]() {
return read_value("v2.txt");
});
std::cout << f1.get() + f2.get() << '\n';
}
Any threads spawned to execute the code of read_value
are hidden behind
the call to WorkQueue::enqueue
. The user code deals only with future
objects, never with raw thread
, promise
, or packaged_task
objects.
Enforcement
???