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

news/2025/2/23 13:57:28

在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.niftyadmin.cn/n/5863452.html

相关文章

LeetCode 2506.统计相似字符串对的数目:哈希表+位运算

【LetMeFly】2506.统计相似字符串对的数目&#xff1a;哈希表位运算 力扣题目链接&#xff1a;https://leetcode.cn/problems/count-pairs-of-similar-strings/ 给你一个下标从 0 开始的字符串数组 words 。 如果两个字符串由相同的字符组成&#xff0c;则认为这两个字符串 …

【多模态处理篇三】【DeepSeek语音合成:TTS音色克隆技术揭秘】

最近帮某明星工作室做AI语音助手时遇到魔幻需求——要求用5秒的咳嗽声克隆出完整音色!传统TTS系统直接翻车,生成的语音像得了重感冒的电音怪物。直到祭出DeepSeek的TTS音色克隆黑科技,才让AI语音从"机器朗读"进化到"声临其境"。今天我们就来扒开这个声音…

【Leetcode 每日一题 - 扩展】1512. 好数对的数目

问题背景 给你一个整数数组 n u m s nums nums。 如果一组数字 ( i , j ) (i,j) (i,j) 满足 n u m s [ i ] n u m s [ j ] nums[i] nums[j] nums[i]nums[j] 且 i < j i < j i<j&#xff0c;就可以认为这是一组 好数对 。 返回好数对的数目。 数据约束 1 ≤ n …

vivado修改下载器下载速率

Error Launching Program X Error while launching program: fpga configuration failed. DONE PIN is not HIGH 原因是下载器速度太快了。先从任务管理器中关闭hw_server.exe试一下,要是不行就按下面三种方法解决。 第一种方法可以不用修改下载速度,直接先从vivado中将bit流…

ESP32-S3 实战指南:BOOT-KEY 按键驱动开发全解析

一、基础知识 本篇我们使用 BOOT 按键来学习一下 GPIO 功能&#xff0c;首先补充一下相关术语介绍。 1、GPIO&#xff08;General Purpose Input/Output&#xff09; GPIO 是微控制器上的通用引脚&#xff0c;既可以作为输入&#xff08;读取外部信号&#xff09;&#xff0…

国产开源PDF解析工具MinerU

前言 PDF的数据解析是一件较困难的事情&#xff0c;几乎所有商家都把PDF转WORD功能做成付费产品。 PDF是基于PostScript子集渲染的&#xff0c;PostScript是一门图灵完备的语言。而WORD需要的渲染&#xff0c;本质上是PDF能力的子集。大模型领域&#xff0c;我们的目标文件格…

C++算法基础笔记

算法学习 C语法字符和字符串输出输出字符串拼接和扩充检查字符串是否存在大写、小写字母字符数组换行 C语法 字符和字符串输出输出 在C 中使用如下语法实现对容器中的对象进行遍历&#xff0c;类似于js或python的for in语法 for (element_declaration : container)#include …

L2-【英音】地道语音语调

文章目录 英音音标节奏节奏感体现词重音句重音音节的划分音节的突出元音单元音/ɑː//ɔ://u://i://ɜː////ʌ//ɒ//ʊ//ɪ//ə//e/双元音/eɪ//aɪ//aʊ//əʊ//ɔɪ//ɪə//ʊə//eə/目标 1.相较于之前有较大进步的语音面貌; 2.自己可以为自己纠正发音; 3.在听到一篇文章…