【C++入门到精通】Lock_guard与Unique_lock C++11 [ C++入门 ]

阅读导航
- 引言
- 一、RAII机制
- 1. 概念
- 2. 原理
- 3. 优点
- 二、Lock_guard
- 1. 官方文档
- 2. 概念
- 3. 底层类模版
- 4. 使用示例
- 三、Unique_lock
- 1. 官方文档
- 2. 概念及底层
- 3. 使用示例
- 四、总结
- 温馨提示
引言
在C++11标准中,为了更方便地使用互斥锁(Mutex)来保证多线程程序的安全性,Lock_guard和Unique_lock这两个类也被引入。它们作为RAII(资源获取即初始化)机制的一种实现方式,在多线程编程中起到了重要的作用。Lock_guard和Unique_lock可以帮助程序员自动管理互斥锁的加锁和解锁,避免忘记解锁而导致的死锁等问题。本文将详细介绍Lock_guard和Unique_lock的使用方法和区别,并通过实例展示如何使用它们来实现线程安全的程序。
一、RAII机制
1. 概念
RAII(Resource Acquisition Is Initialization)是一种C++编程技术,它通过将资源的获取和释放与对象的生命周期绑定在一起,以确保资源在对象创建时获取,在对象销毁时释放。这种技术利用了C++对象的构造函数和析构函数的调用机制,使得资源的管理变得更加简洁、安全和可靠。
2. 原理
使用RAII的关键在于将资源的获取和释放操作分别放置在对象的构造函数和析构函数中。当对象被创建时,构造函数负责获取资源并进行必要的初始化工作;当对象被销毁时,析构函数负责释放资源并进行清理工作。由于C++保证在对象销毁时析构函数会被自动调用,所以资源的释放也就得到了保证。
3. 优点
RAII技术的优点如下:
- 简洁性:通过对象的自动构造和析构,减少了手动管理资源的代码量,使得程序更加简洁易读。
- 安全性:确保资源在适当的时候被释放,避免了常见的资源泄漏和错误状态的产生。
- 可靠性:无论在何时何地发生异常或提前退出,都能够保证资源的正确释放,提高程序的可靠性。
- 可扩展性:通过继承和组合等方式,可以方便地扩展和管理更复杂的资源。
常见的使用RAII技术的例子包括使用智能指针管理动态内存、使用文件对象进行文件操作、使用互斥锁类进行线程同步等。通过合理运用RAII技术,可以有效地提高代码的可维护性、可读性和可靠性,是现代C++编程中的重要技术之一。
二、Lock_guard
1. 官方文档
⭕Lock_guard官方文档

2. 概念
std::lock_guard是C++标准库中的一个RAII类模板,用于管理互斥锁(std::mutex)的加锁和解锁操作。它提供了一种简单且安全的方式来确保在退出作用域时互斥锁会被正确地释放,从而避免了忘记解锁而导致的死锁和资源泄漏等问题。
3. 底层类模版
// lock_guard类模板,用于管理互斥锁的加锁和解锁操作
template<class _Mutex>
class lock_guard
{
public:// 构造函数,在创建lock_guard对象时自动上锁互斥锁_Mtxexplicit lock_guard(_Mutex& _Mtx): _MyMutex(_Mtx){_MyMutex.lock(); // 使用互斥锁的lock()成员函数进行上锁操作}// 构造函数,在已经上锁的情况下创建lock_guard对象// 在这种情况下,不需要再次上锁互斥锁,因此该构造函数空实现lock_guard(_Mutex& _Mtx, adopt_lock_t): _MyMutex(_Mtx){}// 析构函数,在lock_guard对象销毁时自动解锁互斥锁_Mtx// 使用互斥锁的unlock()成员函数进行解锁操作~lock_guard() _NOEXCEPT{_MyMutex.unlock();}// 禁用拷贝构造函数和拷贝赋值运算符,确保lock_guard对象不可拷贝lock_guard(const lock_guard&) = delete;lock_guard& operator=(const lock_guard&) = delete;private:_Mutex& _MyMutex; // 引用类型成员变量,用于保存互斥锁的引用
};
lock_guard类有以下几个重要成员函数:
explicit lock_guard(_Mutex& _Mtx):该构造函数会在创建lock_guard对象时自动上锁互斥锁_Mtx。它使用互斥锁的lock()成员函数进行上锁操作。lock_guard(_Mutex& _Mtx, adopt_lock_t):该构造函数用于在已经上锁的情况下创建lock_guard对象。在这种情况下,不需要再次上锁互斥锁,因此该构造函数空实现。~lock_guard() _NOEXCEPT:析构函数会在lock_guard对象销毁时自动解锁互斥锁_Mtx。它使用互斥锁的unlock()成员函数进行解锁操作。lock_guard(const lock_guard&) = delete和lock_guard& operator=(const lock_guard&) = delete:禁用拷贝构造函数和拷贝赋值运算符,确保lock_guard对象不可拷贝。
🚨注意:lock_guard禁用了拷贝构造函数和拷贝赋值运算符,意味着它不支持拷贝语义,只能通过直接创建对象来使用。这样可以避免多个lock_guard对象同时管理同一个互斥锁而导致的错误行为。
4. 使用示例
使用std::lock_guard非常简单,只需在需要加锁的代码块的开始处创建一个std::lock_guard对象,并将互斥锁作为参数传递给它的构造函数。当代码块结束时,std::lock_guard对象的析构函数会自动调用,从而触发互斥锁的解锁操作。
下面是一个示例代码,展示了如何使用std::lock_guard来管理互斥锁的加锁和解锁:
#include <iostream>
#include <thread>
#include <mutex>std::mutex mtx; // 定义一个互斥锁void printMessage(const std::string& message) {std::lock_guard<std::mutex> lock(mtx); // 创建std::lock_guard对象并传入互斥锁for (int i = 0; i < 5; ++i) {std::cout << message << std::endl;}
}int main() {std::thread t1(printMessage, "Hello");std::thread t2(printMessage, "World");t1.join();t2.join();return 0;
}
在上述示例中,printMessage函数通过创建一个std::lock_guard对象lock,来确保在执行打印操作之前获得互斥锁,并在函数返回时自动释放互斥锁。这样,当多个线程调用printMessage函数时,它们之间的执行将会排他性地进行,避免了数据竞争和输出混乱的问题。
总之,std::lock_guard提供了一种简单且安全的方式来管理互斥锁的加锁和解锁操作,帮助我们在使用互斥锁时避免常见的错误,并提高代码的可靠性和可维护性。
三、Unique_lock
1. 官方文档
⭕Unique_lock官方文档

2. 概念及底层
unique_lock类模板与lock_guard类似,都是以资源获取就是初始化(Resource Acquisition Is Initialization,缩写为RAII)的方式对锁进行封装。它们都采用了独占所有权的机制,即不能进行拷贝操作。在构造或移动赋值时,需要将一个Mutex对象作为参数传递给unique_lock对象,新创建的unique_lock对象会负责管理该Mutex对象的上锁和解锁操作。
对于使用unique_lock实例化的对象,当对象被创建时,会自动调用构造函数对Mutex对象进行上锁操作,确保当前线程获得互斥访问权限。而当unique_lock对象销毁时,会自动调用析构函数解锁Mutex对象,释放互斥访问权限。通过这种方式,可以方便地避免死锁问题的发生,保证线程安全性。
与lock_guard不同的是,unique_lock更加的灵活,提供了更多的成员函数:
- 上锁/解锁操作:
lock、try_lock、try_lock_for、try_lock_until和unlock - 修改操作:移动赋值、交换(
swap():与另一个unique_lock对象互换所管理的互斥量所有权)、释放(release():返回它所管理的互斥量对象的指针,并释放所有权) - 获取属性:
owns_lock(返回当前对象是否上了锁)、operator bool()(与owns_lock()的功能相同)、mutex(返回当前unique_lock所管理的互斥量的指针)。
总的来说unique_lock类模板与lock_guard类似,都提供了一种方便且安全地管理互斥量的方法。它们通过RAII的思想,在对象的生命周期中自动管理锁的上锁和解锁操作,从而简化了编程过程,减少了出错的可能性,并提高了代码的可读性和可维护性。
3. 使用示例
当使用 unique_lock 时,一般需要搭配一个 std::mutex 来进行线程间同步。以下是一个简单的示例:
#include <iostream>
#include <thread>
#include <mutex>std::mutex mtx; // 创建一个互斥量void work_in_critical_section() {std::unique_lock<std::mutex> lock(mtx); // 在进入临界区之前,使用 unique_lock 对互斥量进行上锁操作// 在这里执行需要互斥访问的操作std::cout << "Critical section is locked by this thread" << std::endl;
} // 离开作用域时,unique_lock 的析构函数会自动解锁互斥量int main() {std::thread t1(work_in_critical_section);std::thread t2(work_in_critical_section);t1.join();t2.join();return 0;
}
在这个示例中,我们创建了一个互斥量 mtx,然后在 work_in_critical_section 函数中,我们使用 std::unique_lock 对 mtx 进行上锁操作。在 main 函数中,我们创建了两个线程分别执行 work_in_critical_section 函数。由于使用了 unique_lock,它会在离开作用域时自动解锁互斥量,确保线程安全。
四、总结
RAII机制是一种重要的编程范式,Lock_guard和Unique_lock是C++标准库中用于资源管理的类模板。它们都基于RAII机制,能够简化资源的管理,提高代码的可读性和可维护性。同时,它们也提供了对互斥量的自动上锁和解锁操作,确保了线程安全。
温馨提示
感谢您对博主文章的关注与支持!另外,我计划在未来的更新中持续探讨与本文相关的内容,会为您带来更多关于C++以及编程技术问题的深入解析、应用案例和趣味玩法等。请继续关注博主的更新,不要错过任何精彩内容!
再次感谢您的支持和关注。期待与您建立更紧密的互动,共同探索C++、算法和编程的奥秘。祝您生活愉快,排便顺畅!

相关文章:
【C++入门到精通】Lock_guard与Unique_lock C++11 [ C++入门 ]
阅读导航 引言一、RAII机制1. 概念2. 原理3. 优点 二、Lock_guard1. 官方文档2. 概念3. 底层类模版4. 使用示例 三、Unique_lock1. 官方文档2. 概念及底层3. 使用示例 四、总结温馨提示 引言 在C11标准中,为了更方便地使用互斥锁(Mutex)来保…...
电路设计(8)——计时器的multism仿真
1.功能设计 这是一个计时电路,在秒脉冲的驱动下,计时器开始累加,6个数码管分别显示计时的 时:分:秒。 仿真图如下所示: 左边的运放构成了振荡电路,可以产生脉冲波。这个脉冲波给计时电路提供基准…...
Jmeter测试实践:文件下载接口
一 Jmeter步骤 1.打开jmeter4.0,新建测试计划,添加线程组。根据实际情况配置线程属性。 2.添加HTTP请求。根据接口文档进行配置。 Basic部分修改如下,Advanced部分保持默认。这里的参数id是文件的id,我进行了参数化,…...
PyQt5实现学生管理系统第三天(下)
目录 一:学生课程导航 二:搜索框 三:查询 四:页面数据展示逻辑 上一节,我们介绍了课程管理的课程查询导航的功能。这一节我们介绍下学生课程的功能实现,因为学生课程只有一个查询列表,内容相对简单,所以我们在这一节也重点讲述下我们页面的展现逻辑。 一:学生课程…...
第4章 | 安徽某高校《统计建模与R软件》期末复习
第4章 参数估计 参数估计是统计建模的关键步骤之一,它涉及根据样本数据推断总体参数的过程。在统计学中,参数通常用于描述总体的特征,如均值、方差等。通过参数估计,我们可以利用样本信息对这些未知参数进行推断,从而…...
localforage本地存储(融合Web Storage,Web SQL Database,ndexedDB三种前端存储)
介绍 localForage 是一个快速而简单的 JavaScript 存储库。通过使用异步存储(IndexedDB 或 WebSQL)和简单的类 localStorage 的 API ,localForage 能改善 Web 应用的离线体验。 在不支持 IndexedDB 或 WebSQL 的浏览器中,localF…...
【JavaWeb学习笔记】17 - ThreadLocal
项目代码 https://github.com/yinhai1114/JavaWeb_LearningCode/tree/main/threadlocal/src/com/yinhai/thread 目录 项目代码 一、什么是ThreadLocal? 二、ThreadLocal快速入门 三、源码解读 一、什么是ThreadLocal? 1. ThreadLocal的作用,可以实现在同一个线…...
【ARMv8M Cortex-M33 系列 1 -- SAU 介绍】
文章目录 Cortex-M33 SAU 介绍SAU 的主要功能包括SAU 寄存器配置示例 Cortex-M33 SAU 介绍 在 ARMv8-M 架构中,SAU(Security Attribution Unit)是安全属性单元,用于配置和管理内存区域的安全属性。SAU 是 ARM TrustZone 技术的一…...
sklearn 逻辑回归Demo
逻辑回归案例 假设表示 基于上述情况,要使分类器的输出在[0,1]之间,可以采用假设表示的方法。 设 h θ ( x ) g ( θ T x ) h_θ (x)g(θ^T x) hθ(x)g(θTx), 其中 g ( z ) 1 ( 1 e − z ) g(z)\frac{1}{(1e^{−z} )} g(z)(1e−z)1…...
什么是众创空间?他有什么特点?
众创空间,是一种为大众创新创业提供专业化服务的创业服务平台,是顺应网络时代创新创业特点和需求,通过市场化机制、专业化服务和资本化途径构建的低成本、便利化、全要素、开放式的新型创业服务平台的统称。众创空间包括创客空间、联合办公空…...
什么是数据分析思维
参考 一文学会如何做电商数据分析(附运营分析指标框架) 电子商务该如何做数据分析?如何数据分析入门(从各项指标表象进入) https://www.processon.com/outline/6589838c3129f1550cc69950 数据分析步骤 什么是数据分析…...
利用Milvus Cloud和LangChain构建机器人:一种引人入胜且通俗易懂的方法
一、引言 机器人已经深入我们的日常生活,从家庭服务到工业生产,再到医疗和运输等领域。然而,这些机器人往往需要复杂的算法和数据处理技术才能有效地执行任务。在这个过程中,人工智能(AI)和机器学习&#…...
数据结构-如何实现一个队列?逐步解析与代码示例(超详细)
文章目录 前言1.队列的基本概念2.链表与数组实现队列的区别2.1数据存储结构2.2性能2.3内存使用 3.为什么选择链表实现队列?4.结构定义函数声明 5.核心操作5.1初始化 (QInit)5.2销毁 (QDestroy)5.3入队 (QPush)5.4出队 (QPop) 6.队列的查询操作6.1队首元素 (QueueFro…...
爬虫工作量由小到大的思维转变---<第二十三章 Scrapy开始很快,越来越慢(医病篇)>
诊断篇https://blog.csdn.net/m0_56758840/article/details/135170994?ops_request_misc%257B%2522request%255Fid%2522%253A%2522170333243316800180644102%2522%252C%2522scm%2522%253A%252220140713.130102334.pc%255Fall.%2522%257D&request_id1703332433168001806441…...
.Net7.0 或更高版本 System.Drawing.Common 上传图片跨平台方案
项目升级.Net7.0以后,System.Drawing.Common开关已经被删除,且System.Drawing.Common仅在 Windows 上支持 ,于是想办法将原来上传图片验证文件名和获取图片扩展名方法替换一下,便开始搜索相关解决方案。 .Net6.0文档:…...
【MySQL】InnoDB和MyISAM区别
文章目录 一、索引不同1 InnoDB聚簇索引,MyISAM非聚簇索引1 InnoDB聚簇索引2 MyISAM非聚簇索引 2 InnoDB必须要有主键,MyISAM允许没有主键3 InnoDB支持外键4 InnoDB不支持全文索引5 索引保存位置不同 二、对事物的支持三、存储结构不同四、存储空间不同五…...
3分钟了解安全数据交换系统有什么用!
企业为了保护核心数据安全,都会采取一些措施,比如做网络隔离划分,分成了不同的安全级别网络,或者安全域,接下来就是需要建设跨网络、跨安全域的安全数据交换系统,将安全保障与数据交换功能有机整合在一起&a…...
记录汇川:MODBUS TCP-梯形图
H5U的MODBUS通信不需要编写程序,通过组态MODBUS通信配置表,实现数据通信。 Modbus-TCP 主站即Modbus-TCP客户端,通过Modbus-TCP配置,可最多支持同时与31个 Modbus-TCP服务器(从站)进行通讯。 …...
electron + sqlite3 解决打包后无法写入数据库
前言 window环境。 electron28.0.0 sqlite35.1.6 使用 electron-builder 打包。 本文旨在解决打包后无法写入数据库的问题。 但如果你是打包后无法访问sqlite,且有报错弹窗,不妨也看看本文。 也许是同一种原因。 错误原因分析 打包后无法创建db文件&…...
【uniapp小程序-生成二维码+多个图片文字合并一张图】
<!-- 二维码 --><canvas id"qrcode" canvas-id"qrcode" width"120" ></canvas><!-- 生成带小程序码的分享图片 --><canvas canvas-id"shareCanvas" class"share-canvas"></canvas>#qrc…...
从零实现富文本编辑器#5-编辑器选区模型的状态结构表达
先前我们总结了浏览器选区模型的交互策略,并且实现了基本的选区操作,还调研了自绘选区的实现。那么相对的,我们还需要设计编辑器的选区表达,也可以称为模型选区。编辑器中应用变更时的操作范围,就是以模型选区为基准来…...
3.3.1_1 检错编码(奇偶校验码)
从这节课开始,我们会探讨数据链路层的差错控制功能,差错控制功能的主要目标是要发现并且解决一个帧内部的位错误,我们需要使用特殊的编码技术去发现帧内部的位错误,当我们发现位错误之后,通常来说有两种解决方案。第一…...
论文解读:交大港大上海AI Lab开源论文 | 宇树机器人多姿态起立控制强化学习框架(一)
宇树机器人多姿态起立控制强化学习框架论文解析 论文解读:交大&港大&上海AI Lab开源论文 | 宇树机器人多姿态起立控制强化学习框架(一) 论文解读:交大&港大&上海AI Lab开源论文 | 宇树机器人多姿态起立控制强化…...
华硕a豆14 Air香氛版,美学与科技的馨香融合
在快节奏的现代生活中,我们渴望一个能激发创想、愉悦感官的工作与生活伙伴,它不仅是冰冷的科技工具,更能触动我们内心深处的细腻情感。正是在这样的期许下,华硕a豆14 Air香氛版翩然而至,它以一种前所未有的方式&#x…...
Spring是如何解决Bean的循环依赖:三级缓存机制
1、什么是 Bean 的循环依赖 在 Spring框架中,Bean 的循环依赖是指多个 Bean 之间互相持有对方引用,形成闭环依赖关系的现象。 多个 Bean 的依赖关系构成环形链路,例如: 双向依赖:Bean A 依赖 Bean B,同时 Bean B 也依赖 Bean A(A↔B)。链条循环: Bean A → Bean…...
保姆级教程:在无网络无显卡的Windows电脑的vscode本地部署deepseek
文章目录 1 前言2 部署流程2.1 准备工作2.2 Ollama2.2.1 使用有网络的电脑下载Ollama2.2.2 安装Ollama(有网络的电脑)2.2.3 安装Ollama(无网络的电脑)2.2.4 安装验证2.2.5 修改大模型安装位置2.2.6 下载Deepseek模型 2.3 将deepse…...
Proxmox Mail Gateway安装指南:从零开始配置高效邮件过滤系统
💝💝💝欢迎莅临我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:「storms…...
如何应对敏捷转型中的团队阻力
应对敏捷转型中的团队阻力需要明确沟通敏捷转型目的、提升团队参与感、提供充分的培训与支持、逐步推进敏捷实践、建立清晰的奖励和反馈机制。其中,明确沟通敏捷转型目的尤为关键,团队成员只有清晰理解转型背后的原因和利益,才能降低对变化的…...
elementUI点击浏览table所选行数据查看文档
项目场景: table按照要求特定的数据变成按钮可以点击 解决方案: <el-table-columnprop"mlname"label"名称"align"center"width"180"><template slot-scope"scope"><el-buttonv-if&qu…...
【无标题】湖北理元理律师事务所:债务优化中的生活保障与法律平衡之道
文/法律实务观察组 在债务重组领域,专业机构的核心价值不仅在于减轻债务数字,更在于帮助债务人在履行义务的同时维持基本生活尊严。湖北理元理律师事务所的服务实践表明,合法债务优化需同步实现三重平衡: 法律刚性(债…...
