0

0

C++中lock_guard和unique_lock的区别?(灵活度不同的互斥锁封装)

穿越時空

穿越時空

发布时间:2026-01-18 12:34:17

|

606人浏览过

|

来源于php中文网

原创

必须用 unique_lock 而不能用 lock_guard 的情况包括:延迟加锁、手动控制加解锁时机、配合条件变量 wait()、尝试非阻塞加锁(try_lock)、分支逻辑决定是否加锁,以及需要移动语义转移锁所有权。

c++中lock_guard和unique_lock的区别?(灵活度不同的互斥锁封装)

什么时候必须用 unique_lock 而不能用 lock_guard

当需要延迟加锁、手动控制加锁/解锁时机、或配合条件变量使用时,lock_guard 完全无法胜任——它在构造时强制加锁、析构时强制解锁,没有中间状态。unique_lock 支持默认构造(不持有锁)、lock()/unlock() 显式调用、以及 try_lock() 等操作。

  • 条件变量的 wait() 必须传入 unique_lock,因为 wait() 内部要先临时释放锁再挂起线程,唤醒后重新加锁——lock_guard 不支持临时解锁
  • 想尝试加锁但不想阻塞?unique_lock 提供 try_lock()lock_guard 没有对应能力
  • 函数内部分支逻辑才决定是否加锁?只能用 unique_lock 默认构造,后续按需 lock()

unique_lock 的移动语义和所有权转移是干什么用的

unique_lock 可被移动(move),意味着能将锁的所有权从一个对象转移到另一个对象,比如作为函数返回值、存入容器、或跨作用域传递。而 lock_guard 是不可复制也不可移动的,生命周期严格绑定在当前作用域。

  • 常见场景:写一个封装了锁逻辑的 RAII 类,内部持有 unique_lock 成员,靠移动语义把锁“带出去”
  • 错误用法:lock_guard 尝试 std::move 或赋值会编译失败,报错类似 use of deleted function
  • 注意:移动后原 unique_lock 变为空状态(owns_lock() == false),不能再 unlock()

性能和内存开销差异真的存在吗

是的。lock_guard 是零开销抽象:它只保存一个 mutex*,无额外成员,大小通常等于指针宽度;unique_lock 多一个布尔标记位(记录是否持有锁),并需支持延迟构造与移动,编译器优化后仍略大一点,且某些操作(如 try_lock())有轻微运行时分支判断。

  • 纯保护临界区且无需灵活控制?优先选 lock_guard —— 更轻量、意图更明确
  • 不需要移动、也不需要延迟/手动解锁?用 lock_guard 反而更安全:杜绝误操作导致的死锁或未加锁访问
  • 现代编译器对两者的内联和优化都很强,性能差距在绝大多数场景下可忽略,但语义清晰度差异显著

一个典型误用:在 if 分支里用 lock_guard 却以为能控制加锁范围

下面这段代码看似“只在条件成立时加锁”,实际并非如此:

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

Replit Agent
Replit Agent

Replit最新推出的AI编程工具,可以帮助用户从零开始自动构建应用程序。

下载
if (condition) {
    std::lock_guard guard(mtx);
    // 临界区
}

它确实只在 condition 为真时构造 lock_guard,但问题在于:这个 guard 的作用域仍是该 if 块,不是整个函数。很多人真正想要的是“根据条件决定是否进入临界区”,但又希望锁的生命周期跨越多个语句甚至函数调用——这时必须用 unique_lock 手动管理:

std::unique_lock guard;
if (condition) {
    guard = std::unique_lock(mtx); // 延迟加锁
}
// 后续代码可依赖 guard.owns_lock() 判断是否已加锁

否则,仅靠 lock_guard 的作用域机制,无法表达“加锁 → 做事 → 条件性解锁”这类非对称控制流。

最常被忽略的一点:unique_lock 的灵活性是一把双刃剑——它允许你写出正确但难维护的锁逻辑,也容易写出忘记 unlock() 或重复 unlock() 的 bug;而 lock_guard 的刚性恰恰是它的健壮性来源。选哪个,本质是在「控制力」和「约束力」之间做取舍。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

746

2023.08.22

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

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

481

2023.08.10

function是什么
function是什么

function是函数的意思,是一段具有特定功能的可重复使用的代码块,是程序的基本组成单元之一,可以接受输入参数,执行特定的操作,并返回结果。本专题为大家提供function是什么的相关的文章、下载、课程内容,供大家免费下载体验。

478

2023.08.04

js函数function用法
js函数function用法

js函数function用法有:1、声明函数;2、调用函数;3、函数参数;4、函数返回值;5、匿名函数;6、函数作为参数;7、函数作用域;8、递归函数。本专题提供js函数function用法的相关文章内容,大家可以免费阅读。

163

2023.10.07

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

43

2026.01.16

全民K歌得高分教程大全
全民K歌得高分教程大全

本专题整合了全民K歌得高分技巧汇总,阅读专题下面的文章了解更多详细内容。

84

2026.01.16

C++ 单元测试与代码质量保障
C++ 单元测试与代码质量保障

本专题系统讲解 C++ 在单元测试与代码质量保障方面的实战方法,包括测试驱动开发理念、Google Test/Google Mock 的使用、测试用例设计、边界条件验证、持续集成中的自动化测试流程,以及常见代码质量问题的发现与修复。通过工程化示例,帮助开发者建立 可测试、可维护、高质量的 C++ 项目体系。

24

2026.01.16

java数据库连接教程大全
java数据库连接教程大全

本专题整合了java数据库连接相关教程,阅读专题下面的文章了解更多详细内容。

35

2026.01.15

Java音频处理教程汇总
Java音频处理教程汇总

本专题整合了java音频处理教程大全,阅读专题下面的文章了解更多详细内容。

16

2026.01.15

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

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

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