c++怎么处理多线程死锁_c++ lock_guard与unique_lock锁管理【技巧】
技术百科
冰火之心
发布时间:2026-01-01
浏览: 次 死锁本质是资源获取顺序不一致,解决关键是统一加锁顺序;优先用std::scoped_lock处理多锁,单锁用std::lock_guard,需灵活控制时用std::unique_lock配合条件变量。
死锁根本不是锁的问题,而是资源获取顺序不一致
死锁在 C++ 多线程中几乎总是因为多个线程以不同顺序请求同一组互斥量(std::mutex)导致。比如线程 A 先锁 mtx_a 再锁 mtx_b,而线程 B 反过来先锁 mtx_b 再锁 mtx_a——只要两个线程执行节奏稍有交错,就卡死。
解决思路不是换锁类型,而是统一加锁顺序。常见做法包括:
- 给所有互斥量定义全局唯一序号,按序号从小到大加锁(用
std::scoped_lock可自动完成) - 避免在持有锁期间调用可能获取其他锁的函数(尤其是第三方或虚函数)
- 绝不手动调用
lock()/unlock();裸调用是死锁温床
std::lock_guard 适合“进作用域即锁,出作用域即放”的简单场景
std::lock_guard 是最轻量、最安全的 RAII 锁包装器,构造时立即加锁,析构时必然释放,不可转移、不可复制、不可延迟加锁。
它适用于:单个 mutex 的短临界区、不需要条件等待、不涉及多个锁的同步。
std::mutex mtx;
void safe_update() {
std::lock_guard guard(mtx); // 构造即 lock()
// ... 临界区操作
} // 出作用域,guard 析构,自动 unlock()
注意:lock_guard 不支持 try_lock()、不支持 unlock() 提前释放、不能用于 std::condition_variable::wait() ——这些都得换 std::unique_lock。
std::unique_lock 是灵活但需更谨慎的锁管理器
std::unique_lock 支持延迟加锁、手动解锁、条件变量配合、可移动(用于返回锁、传入函数),但灵活性带来责任:忘记 lock() 或重复 unlock() 会引发未定义行为。
典型误用:
- 声明
后没调 lock()就进临界区 → 数据竞争 - 调了
unlock()后又让其析构 → 二次 unlock → UB - 传给
wait()后没检查条件就继续用被临时释放的锁 → 逻辑错乱
正确用法示例(配合条件变量):
std::mutex mtx;
std::condition_variable cv;
bool ready = false;
// 等待方
void wait_for_ready() {
std::unique_lock ulock(mtx);
cv.wait(ulock, []{ return ready; }); // wait 内部会 unlock + 唤醒后重新 lock
// 此处 ulock 已重新持有 mtx,可安全访问 shared data
}
// 通知方
void set_ready() {
std::lock_guard guard(mtx);
ready = true;
cv.notify_one();
}
优先用 std::scoped_lock 解决多锁死锁
C++17 引入的 std::scoped_lock 是处理多个互斥量的首选:它原子性地获取所有锁,内部自动按地址排序(或使用 ADL std::lock 协议),彻底规避因加锁顺序不一致导致的死锁。
对比 lock_guard(只支持一个锁)和手写多 lock()(易出错),scoped_lock 更简洁可靠:
std::mutex mtx_a, mtx_b;
void transfer(int amount) {
// 安全:自动避免死锁
std::scoped_lock lock(mtx_a, mtx_b);
// ... 同时操作两个资源
}
注意:scoped_lock 构造失败(如某 mutex 不可 lock)会抛 std::system_error,且不提供 try_lock 变体——如需非阻塞,应改用 std::try_to_lock_t + unique_lock 组合。
真正容易被忽略的是:即使用了 scoped_lock,如果临界区内又间接触发了其他锁(比如调用了一个你没看源码的库函数),死锁依然可能发生。锁管理只是工具,资源访问契约才是关键。
# ai
# 的是
# 才是
# 尤其是
# 多个
# 适用于
# 不需要
# 不支持
# 工具
# c++
# 线程
# 死锁
# red
# 多线程
# 作用域
# 虚函数
# 加锁
# 有锁
# 一加
# 互斥
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- PythonPandas数据分析项目教程_时间序列
- Win11怎么关闭资讯和兴趣_Windows11任
- 如何在Golang中修改数组元素_通过指针实现原地
- Win11怎么设置系统还原_Windows11系统
- Win11怎么关闭OneDrive同步_Win11
- Windows10蓝屏SYSTEM_SERVICE
- Win10怎么查看内存时序参数_Win10CPU-
- Python文本编码与解码_跨平台解析说明【指导】
- 如何使用Golang搭建Web开发环境_快速启动H
- php485能和物联网模块通信吗_php485对接
- 如何更改Windows资源管理器的默认启动位置?(
- 本地php环境出现502错误_nginx或apac
- Win11怎么设置触控板手势_Windows11三
- 如何优化Golang Web性能_Golang H
- 如何使用 Selenium 正确获取篮球参考网站球
- php8.4新语法match怎么用_php8.4m
- c++中explicit(bool)的用法 c++
- 如何在Golang中实现并发消息队列消费者_Gol
- 微信短链接怎么还原php_用浏览器开发者工具抓包获
- Windows10怎么查看系统激活状态_Windo
- Mac如何解压zip和rar文件?(推荐免费工具)
- Win11怎么关闭任务栏小图标_Windows11
- Python包结构设计_大型项目组织解析【指导】
- Win11怎么设置默认图片查看器_Windows1
- Win10怎样设置多显示器_Win10多显示器扩展
- 作用域操作符会影响性能吗_php静态调用性能分析【
- Win11怎样安装网易云音乐_Win11安装网易云
- c++如何使用std::bitset进行位图算法_
- Windows10无法识别USB设备描述符请求失败
- Golang如何避免指针逃逸_Golang逃逸分析
- php删除数据怎么加限制_带where条件删除避免
- Win11如何设置系统语言_Win11系统语言切换
- c++如何打印函数堆栈信息_c++ backtra
- php本地部署支持nodejs吗_php与node
- Windows家庭版如何开启组策略(gpedit.
- C++如何解析JSON数据?(nlohmann/j
- 如何使用Golang包导出规则_控制函数和变量可见
- Win11怎么清理C盘OneDrive缓存_Win
- Go 语言标准库为何不提供泛型 Contains
- windows 10应用商店区域怎么改_windo
- 如何使用Golang管理模块版本_Golanggo
- Win11文件夹预览图不显示怎么办_Win11缩略
- Win11怎么查看激活状态_查询Windows 1
- 如何用正则与预处理高效拦截带干扰符的恶意域名
- Win11怎么开启剪贴板历史记录_Windows1
- Drupal 中渲染节点时出现 HTML 标签嵌套
- Win11怎么设置虚拟桌面 Win11新建多桌面切
- Python与MongoDB NoSQL开发实战_
- php打包exe后无法读取环境变量_变量配置方法【
- Win10怎么卸载剪映_Win10彻底卸载剪映方法

后没调
QQ客服