【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…...
Qwen2.5-VL-7B-Instruct开源大模型:支持中文优先的多模态理解部署方案
Qwen2.5-VL-7B-Instruct开源大模型:支持中文优先的多模态理解部署方案 1. 项目概述 Qwen2.5-VL-7B-Instruct是一款开源的视觉-语言多模态大模型,特别针对中文场景进行了优化。该模型能够同时处理图像和文本输入,实现跨模态的理解与生成任务…...
HarmonyOS 音乐播放器进阶实战——AVPlayer状态管理与播放列表
1. AVPlayer状态机深度解析 在HarmonyOS音乐播放器开发中,AVPlayer的状态管理就像驾驶手动挡汽车——你需要清楚知道当前处于哪个档位,才能平稳切换。我曾在项目中因为状态处理不当导致音乐卡顿,后来才发现是状态机流转出了问题。 AVPlayer…...
Android tinyalsa深度解析之pcm_params_get_periods_min调用流程与实战(一百七十三)
简介: CSDN博客专家、《Android系统多媒体进阶实战》作者 博主新书推荐:《Android系统多媒体进阶实战》🚀 Android Audio工程师专栏地址: Audio工程师进阶系列【原创干货持续更新中……】🚀 Android多媒体专栏地址&a…...
西门子PLC存储区全解析:从M区到DB块的实战避坑指南
西门子PLC存储区全解析:从M区到DB块的实战避坑指南 第一次接触西门子PLC编程时,面对M区、L区和DB块这些存储区概念,我完全摸不着头脑。记得有次调试设备,明明在线监控看到M0.0已经置位,但程序就是不执行相应动作&#…...
35AE92 GJR5137200R0005电子模块
35AE92 GJR5137200R0005 电子模块是一款工业控制系统用的电子控制模块,通常用于西门子或ABB等自动化设备中,承担信号处理、控制逻辑执行及系统接口功能。开头:35AE92 GJR5137200R0005电子模块是工业自动化控制系统的重要组成部分,…...
5分钟成为效率大师!NoteGen快捷键可视化配置终极指南
5分钟成为效率大师!NoteGen快捷键可视化配置终极指南 【免费下载链接】note-gen 一款专注于记录和写作的跨端 AI 笔记应用。 项目地址: https://gitcode.com/GitHub_Trending/no/note-gen NoteGen是一款专注于记录和写作的跨端AI笔记应用,通过快捷…...
强化学习实战:Sarsa vs Q-learning,on-policy和off-policy到底怎么选?
强化学习实战:Sarsa与Q-learning的深度对比与策略选择指南 1. 理解策略分类的核心逻辑 在强化学习领域,策略选择直接影响算法的行为模式和学习效果。我们先从最基础的概念切入:什么是策略?简单来说,策略就是智能体在特…...
Graphormer实战:预测药物溶解度与渗透性,助力ADMET性质评估
Graphormer实战:预测药物溶解度与渗透性,助力ADMET性质评估 1. 药物研发中的ADMET挑战 在药物研发领域,ADMET(吸收、分布、代谢、排泄和毒性)性质评估是决定候选药物成败的关键环节。传统实验方法耗时耗力࿰…...
避坑指南:GF-3 SAR数据预处理中常见的5个错误及解决方法
GF-3 SAR数据预处理实战:5个关键错误分析与Python解决方案 在遥感数据处理领域,GF-3卫星的合成孔径雷达(SAR)数据因其全天候、全天时的观测能力而备受青睐。然而,从原始数据到可用成果的预处理过程中,即便是经验丰富的工程师也常会…...
韦东山T113工业板+7寸RGB屏保姆级调试笔记:从设备树修改到触摸背光全搞定
T113工业板7寸RGB屏实战调试指南:从设备树到触摸背光的全链路避坑 拿到韦东山T113工业板和配套7寸RGB电容屏的那一刻,很多开发者会迫不及待地开始调试,但很快就会发现事情没那么简单——屏幕不亮、触摸失灵、背光异常等问题接踵而至。本文将带…...
