借用Howard Hinnant 的示例并将其修改为使用复制和交换,这 op= 线程安全吗?
struct A {
A() = default;
A(A const &x); // Assume implements correct locking and copying.
A& operator=(A x) {
std::lock_guard<std::mutex> lock_data (_mut);
using std::swap;
swap(_data, x._data);
return *this;
}
private:
mutable std::mutex _mut;
std::vector<double> _data;
};
我相信这个线程安全(记住 op= 的参数是按值传递的),而我能找到的唯一问题是隐藏在地毯下的问题:复制 ctor。但是,这将是一个罕见的类,它允许复制分配但不允许复制构造,因此该问题同样存在于两种替代方案中。
鉴于自我分配是如此罕见(至少对于这个例子来说),如果它发生我不介意额外的副本,考虑这个 != &rhs 的潜在优化可以忽略不计或悲观化。与原始策略(下)相比,是否还有其他理由更喜欢或避免它?
A& operator=(A const &rhs) {
if (this != &rhs) {
std::unique_lock<std::mutex> lhs_lock( _mut, std::defer_lock);
std::unique_lock<std::mutex> rhs_lock(rhs._mut, std::defer_lock);
std::lock(lhs_lock, rhs_lock);
_data = rhs._data;
}
return *this;
}
顺便说一句,我认为这简洁地处理了复制ctor,至少对于这个类,即使它有点迟钝:
A(A const &x) : _data {(std::lock_guard<std::mutex>(x._mut), x._data)} {}