首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >在两个互斥锁上获取锁,避免死锁

在两个互斥锁上获取锁,避免死锁
EN

Stack Overflow用户
提问于 2010-02-12 11:53:35
回答 6查看 7.2K关注 0票数 10

下面的代码包含一个潜在的死锁,但似乎是必要的:要安全地将数据从一个容器复制到另一个容器,必须锁定两个容器,以防止在另一个线程中发生更改。

代码语言:javascript
复制
void foo::copy(const foo & rhs)
{
    pMutex->lock();
    rhs.pMutex->lock();
    // do copy
}

Foo有一个STL容器,"do copy“本质上就是使用std::copy。如何在不引入死锁的情况下锁定两个互斥锁?

EN

回答 6

Stack Overflow用户

回答已采纳

发布于 2010-02-12 11:55:09

foo实例施加某种总顺序,并始终以递增或递减顺序获取它们的锁,例如,先是foo1->lock(),然后是foo2->lock()

另一种方法是使用函数语义,而不是编写一个创建新实例的foo::clone方法,而不是破坏现有实例。

如果您的代码执行大量锁定,则可能需要一个复杂的死锁避免算法,如banker's algorithm

票数 16
EN

Stack Overflow用户

发布于 2010-02-12 12:27:59

这个怎么样?

代码语言:javascript
复制
void foo::copy(const foo & rhs)
{
    scopedLock lock(rhs.pMutex); // release mutex in destructor
    foo tmp(rhs);
    swap(tmp); // no throw swap locked internally
}

这是异常安全的,也是线程安全的。为了100%节省线程,你需要检查所有的代码路径,然后用另一组眼睛再次检查,然后再次检查它……

票数 1
EN

Stack Overflow用户

发布于 2020-10-12 17:48:45

正如@Mellester提到的,你可以使用std::lock来锁定多个互斥锁,避免死锁。

代码语言:javascript
复制
#include <mutex>

void foo::copy(const foo& rhs)
{
    std::lock(pMutex, rhs.pMutex);

    std::lock_guard<std::mutex> l1(pMutex, std::adopt_lock);
    std::lock_guard<std::mutex> l2(rhs.pMutex, std::adopt_lock);

    // do copy
}

但请注意,要检查rhs不是*this,因为在这种情况下,由于锁定同一互斥锁,std::lock将导致UB。

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/2249571

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档