0

0

C++如何利用std::lock_guard和std::scoped_lock避免死锁?(并发安全)

穿越時空

穿越時空

发布时间:2026-03-05 14:37:31

|

506人浏览过

|

来源于php中文网

原创

c++如何利用std::lock_guard和std::scoped_lock避免死锁?(并发安全)

std::scoped_lock 能自动避免死锁,但必须一次性锁定所有互斥量

死锁常发生在多个线程以不同顺序调用 std::lock_guard 锁定多个 std::mutex 时。比如线程 A 先锁 mtx1 再锁 mtx2,线程 B 反过来先锁 mtx2 再锁 mtx1 —— 这种场景下 std::lock_guard 完全无法预防死锁,它只负责单个互斥量的 RAII 管理。

std::scoped_lock 的核心价值是:在构造时对传入的所有互斥量调用 std::lock(底层使用死锁避免算法,如尝试加锁 + 退避重试),确保要么全部成功,要么一个都不上锁。这从源头切断了“部分加锁、等待对方释放”的死锁路径。

常见错误是把它当 std::lock_guard 的多参数替代品,却分多次构造:

// ❌ 错误:两次独立构造,仍可能死锁
std::scoped_lock l1(mtx1);
std::scoped_lock l2(mtx2); // 此时 mtx2 可能已被别的线程持有了,而 mtx1 已锁住

// ✅ 正确:一次性声明所有要锁的互斥量
std::scoped_lock l(mtx1, mtx2); // 原子性获取两个锁
  • 必须把所有需要的 std::mutex(或支持 lock()/try_lock()/unlock() 的类型)一次性传给 std::scoped_lock 构造函数
  • 不支持运行时动态决定锁哪些互斥量(比如根据条件选锁 1 个或 2 个),这种需求得回退到 std::lock + 手动 std::lock_guard
  • 注意兼容性:std::scoped_lock 是 C++17 引入的,C++14 及更早只能用 std::lock + std::adopt_lock

std::lock_guard 本身不防死锁,但配合 lock/unlock 顺序约定可降低风险

很多人误以为只要用了 std::lock_guard 就“线程安全”了,其实它只保证单个锁的自动释放,对多锁顺序毫无约束。真正防死锁靠的是工程规范,而非这个类本身。

立即学习C++免费学习笔记(深入)”;

典型错误现象:程序偶发卡死,gdb 查看线程状态显示多个线程都停在 pthread_mutex_lockstd::mutex::lock(),且各自持有某锁、等待另一锁 —— 这就是典型的循环等待型死锁。

AskManyAI
AskManyAI

AskManyAI是个一站式AI聚合平台,集成了国内外多个主流顶尖AI大模型

下载
  • 所有模块必须约定全局一致的锁顺序,例如“总是先锁 g_config_mutex,再锁 g_cache_mutex”,并在注释和代码审查中强制检查
  • 避免在持有锁期间调用用户可扩展的回调(如 std::function)、虚函数或第三方库接口,这些可能间接触发未知的锁请求
  • 不要在析构函数里试图获取另一把锁 —— 析构时机不可控,极易打破锁顺序

std::scoped_lock 的性能开销比 std::lock_guard 略高,但通常可忽略

std::scoped_lock 构造时需协调多个互斥量,内部可能调用多次 try_lock() 并做指数退避,相比单锁的 std::lock_guard 多几条指令和一次潜在的短暂自旋。但在绝大多数实际负载下,这点差异远小于锁本身的竞争开销。

真正影响性能的是锁粒度和持有时间,不是选哪个 RAII 类型。别为了省几个 CPU 周期而放弃死锁防护。

  • 如果只锁一个互斥量,std::lock_guardstd::scoped_lock 行为一致,性能无差别
  • 若锁两个以上互斥量,std::scoped_lock 的死锁避免逻辑会带来微小但确定的额外成本;不过比起死锁导致的服务不可用,这点成本几乎不值一提
  • 不要手动实现“先 try_lock 再 fallback”来模拟 std::scoped_lock —— 标准库的实现已针对常见平台做了优化,自己写容易漏掉边界情况

std::scoped_lock 不解决嵌套锁、条件竞争或逻辑错误

它只管“多锁同时获取时不发生死锁”,其他并发问题一概不处理。比如两个线程都用 std::scoped_lock(mtx1, mtx2),但一个读 data1data2,另一个读 data2data1,照样可能因数据依赖产生逻辑错误。

还有更隐蔽的坑:std::scoped_lock 不能跨作用域传递,也不能复制。下面这段代码编译失败,但新手常以为能“转移锁权”:

// ❌ 编译错误:std::scoped_lock 不可复制
std::scoped_lock l(mtx1, mtx2);
auto l2 = std::move(l); // 即使 move 也不行,它没有移动构造函数
  • 锁的作用域必须清晰,避免在 lambda 捕获后延长生命周期,尤其不能捕获到异步任务里
  • 它不阻止数据竞争 —— 如果你忘了用锁保护某个共享变量的访问,std::scoped_lock 不会报错也不会警告
  • 异常安全没问题(构造失败则不持有任何锁,成功则析构自动释放),但别指望它帮你发现遗漏的临界区

最麻烦的从来不是怎么加锁,而是判断“这里到底该不该加锁”“该锁哪几个变量”“有没有漏掉某个并发修改点”。标准库工具只执行指令,不理解业务逻辑。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

214

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

192

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

60

2026.01.05

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1800

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

593

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2341

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

45

2026.01.19

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

743

2023.08.10

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

2

2026.03.05

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 10.6万人学习

C 教程
C 教程

共75课时 | 5.2万人学习

C++教程
C++教程

共115课时 | 20.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号