Is there a C++11 equivalent for the boost::shared_mutex
. Or another solution to handle a multiple reader / single writer situation in C++11?
问题:
回答1:
I tried but failed to get shared_mutex
into C++11. It has been proposed for a future standard. The proposal is here.
Edit: A revised version (N3659) was accepted for C++14.
Here is an implementation:
http://howardhinnant.github.io/shared_mutex
http://howardhinnant.github.io/shared_mutex.cpp
回答2:
Simple... There isn't one. There is no standard C++ implementation of a readers-writer lock.
But, you have a few options here.
- You are left at your own devices to make your own readers-writer lock.
- Use a platform-specific implementation such as Win32's, POSIX's, or Boost's as you mention.
- Don't use one at all -- use a mutex which already exists in C++11.
Going with #1 and implementing your own is a scary undertaking and it is possible to riddle your code with race conditions if you don't get it right. There is a reference implemenation that may make the job a bit easier.
If you want platform independent code or don't want to include any extra libraries in your code for something as simple as a reader-writer lock, you can throw #2 out the window.
And, #3 has a couple caveats that most people don't realize: Using a reader-writer lock is often less performant, and has more difficult-to-understand code than an equivalent implementation using a simple mutex. This is because of the extra book-keeping that has to go on behind the scenes of a readers-writer lock implementation.
I can only present you your options, really it is up to you to weigh the costs and benefits of each and pick which works best.
Edit:
C++17 now has a shared_mutex
type for situations where the benefits of having multiple concurrent readers outweigh the performance cost of the shared_mutex
itself.
回答3:
No, there is no equivalent for boost::shared_mutex
in C++11.
Read/writer locks are supported in C++14 or later, though:
- C++14 added
std::shared_timed_mutex
- C++17 added
std::shared_mutex
The difference is that std::shared_timed_mutex
adds additional timing operations. It implements the SharedTimedMutex concept, which is an extension of the simpler TimedMutex concept implemented by std::shared_mutex
.
Keep in mind that acquiring a lock for a read/writer mutex is more costly than acquiring a normal std::mutex
. As a consequence, a read/writer mutex will not improve the performance if you have frequent, but short read operations. It is better suited for scenarios were read operations are frequent and expensive. To quote from Anthony Williams' post:
The cost of locking a shared_mutex is higher than that of locking a plain std::mutex, even for the reader threads. This is a necessary part of the functionality --- there are more possible states of a shared_mutex than a mutex, and the code must handle them correctly. This cost comes in both the size of the object (which in both your implementation and my POSIX implementation includes both a plain mutex and a condition variable), and in the performance of the lock and unlock operations.
Also, the shared_mutex is a point of contention, and thus not scalable. Locking a shared_mutex necessarily modifies the state of the mutex, even for a read lock. Consequently, the cache line holding the shared_mutex state must be transferred to whichever processor is performing a lock or unlock operation.
If you have a lot of threads performing frequent, short read operations, then on a multiprocessor system this can lead to a lot of cache ping-pong, which will considerably impact the performance of the system. In this case, you may as well adopt the simpler design of just using a plain mutex, as the readers are essentially serialized anyway.
If the reads are not frequent, then there is no contention, so you don't need to worry about concurrent readers, and a plain mutex will suffice for that scenario anyway.
If the read operations are time consuming, then the consequence of this contention is less visible, since it is dwarfed by the time spent whilst holding the read lock. However, performing time consuming operations whilst holding a lock is a design smell.
In the vast majority of cases, I think that there are better alternatives to a shared_mutex. These may be a plain mutex, the atomic support of shared_ptr, the use of a carefully constructed concurrent container, or something else, depending on context.