当前位置: 首页 > article >正文

std::lock_guard、std::unique_lock、std::shared_lock

在C++中,std::lock_guardstd::unique_lockstd::shared_lock是用于管理互斥量的RAII类,确保锁的正确获取和释放(避免忘记释放锁导致的死锁问题)。以下是它们的详细介绍、区别及使用场景:


1. std::lock_guard

  • 功能:最简单的锁管理器,构造时立即锁定互斥量,析构时自动释放。
  • 特点
    • 不支持手动锁定或解锁。
    • 不支持延迟锁定或条件变量。
    • 不可复制或移动,仅限当前作用域使用。
  • 适用场景
    • 保护临界区,无需中途解锁或额外灵活性。
  • 示例
    std::mutex mtx;
    {
        std::lock_guard<std::mutex> lock(mtx); // 立即锁定
        // 临界区操作
    } // 自动解锁
    

2. std::unique_lock

  • 功能:灵活的锁管理器,支持手动控制锁,适用于复杂场景。
  • 特点
    • 可延迟锁定(defer_lock)、尝试锁定(try_to_lock)或接管已锁定互斥量(adopt_lock)。
    • 允许手动调用lock()unlock()
    • 支持所有权转移(移动语义)。
    • 必须与条件变量配合使用。
  • 适用场景
    • 需要延迟锁定、中途解锁或条件变量。
    • 锁的所有权需在不同作用域传递。
  • 示例
    std::mutex mtx;
    std::unique_lock<std::mutex> lock(mtx, std::defer_lock); // 延迟锁定
    lock.lock(); // 手动锁定
    // ... 临界区操作
    lock.unlock(); // 手动解锁(可选)
    

std::unique_lock 提供了以下几个成员函数:

  • lock():尝试对互斥量进行加锁操作,如果当前互斥量已经被其他线程持有,则当前线程会被阻塞,直到互斥量被成功加锁。

  • try_lock():尝试对互斥量进行加锁操作,如果当前互斥量已经被其他线程持有,则函数立即返回 false,否则返回 true。

  • try_lock_for(const std::chrono::duration<Rep, Period>& rel_time):尝试对互斥量进行加锁操作,如果当前互斥量已经被其他线程持有,则当前线程会被阻塞,直到互斥量被成功加锁,或者超过了指定的时间。

必须要用时间互斥锁。std::timed_mutex mtx;

try_lock_until(const std::chrono::time_point<Clock, Duration>& abs_time):尝试对互斥量进行加锁操作,如果当前互斥量已经被其他线程持有,则当前线程会被阻塞,直到互斥量被成功加锁,或者超过了指定的时间点。

unlock():对互斥量进行解锁操作。

除了上述成员函数外,std::unique_lock 还提供了以下几个构造函数:

unique_lock() noexcept = default:默认构造函数,创建一个未关联任何互斥量的 std::unique_lock 对象。

explicit unique_lock(mutex_type& m):构造函数,使用给定的互斥量 m 进行初始化,并对该互斥量进行加锁操作。

unique_lock(mutex_type& m, defer_lock_t) noexcept:构造函数,使用给定的互斥量 m 进行初始化,但不对该互斥量进行加锁操作。

unique_lock(mutex_type& m, try_to_lock_t) noexcept:构造函数,使用给定的互斥量 m 进行初始化,并尝试对该互斥量进行加锁操作。如果加锁失败,则创建的 std::unique_lock 对象不与任何互斥量关联。

unique_lock(mutex_type& m, adopt_lock_t) noexcept:构造函数,使用给定的互斥量 m 进行初始化,并假设该互斥量已经被当前线程成功加锁。


3. std::shared_lock

  • 功能:管理共享互斥量(如std::shared_mutex)的共享锁,允许多线程并发读。
  • 特点
    • 构造时以共享模式锁定(lock_shared())。
    • 支持延迟锁定、手动操作,类似std::unique_lock
    • 析构时自动释放共享锁。
  • 适用场景
    • 多线程读取共享数据,需并发访问。
    • 写入时需配合std::unique_lockstd::lock_guard进行独占锁定。
  • 示例
    std::shared_mutex sh_mtx;
    {
        std::shared_lock<std::shared_mutex> read_lock(sh_mtx); // 共享锁定
        // 多线程可并发读取
    } // 自动释放共享锁
    
    {
        std::lock_guard<std::shared_mutex> write_lock(sh_mtx); // 独占锁定
        // 单线程写入
    }
    

关键区别总结

特性std::lock_guardstd::unique_lockstd::shared_lock
锁定模式独占独占共享
灵活性低(自动锁定/解锁)高(手动控制)中(类似unique_lock
延迟锁定不支持支持(defer_lock支持(defer_lock
条件变量支持不支持支持不支持
适用互斥量类型普通互斥量(如std::mutex普通互斥量共享互斥量(如std::shared_mutex
性能开销较高中等

选择建议

  • 简单临界区:优先用std::lock_guard,轻量且高效。
  • 复杂控制(如条件变量):用std::unique_lock
  • 多读单写场景:读用std::shared_lock,写用std::unique_lockstd::lock_guard

通过合理选择锁管理器,可以避免死锁并提升多线程程序的性能和可靠性。


我悠哉悠哉邀请我的灵魂,弯腰闲看一片夏天的草叶。 —沃尔特·惠特曼


http://www.kler.cn/a/557095.html

相关文章:

  • 青少年软件编程(C语言)等级三级考试试题(2)
  • DeepSeek 到底是什么类型的应用,其核心功能是什么?
  • 工业机器人中用于3D碰撞检测的算法库有哪些
  • ubuntu ffmpeg 安装踩坑
  • 【Python项目】基于Django的网站验证码的生成与识别系统
  • 基于Java+SpringBoot+Vue的前后端分离的汽车租赁系统
  • 正则表达式效验邮箱格式, 手机号格式, 密码长度
  • Python 学习之旅:高级阶段(十四)Web 开发框架 Flask
  • 邮件安全之发件人伪造
  • [漏洞篇]文件上传漏洞详解
  • matlab 车辆进出检测算法设计GUI界面-论文
  • 蓝桥杯刷题2.21|笔记
  • 爬虫学习第八篇-学习小总结
  • MYSQL查询优化器选错了索引怎么办
  • matlab 轮边驱动系统汽车垂向动力学分析
  • python学opencv|读取图像(七十四)人脸识别:EigenFaces算法
  • vue中将el-table导出为excel文件
  • 使用LangChain构建第一个ReAct Agent
  • C/C++ | 每日一练 (2)
  • 大模型掀起AI新浪潮,传感器成为智能升级关键