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

【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技术的优点如下:

  1. 简洁性:通过对象的自动构造和析构,减少了手动管理资源的代码量,使得程序更加简洁易读。
  2. 安全性:确保资源在适当的时候被释放,避免了常见的资源泄漏和错误状态的产生。
  3. 可靠性:无论在何时何地发生异常或提前退出,都能够保证资源的正确释放,提高程序的可靠性。
  4. 可扩展性:通过继承和组合等方式,可以方便地扩展和管理更复杂的资源。

常见的使用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类有以下几个重要成员函数:

  1. explicit lock_guard(_Mutex& _Mtx):该构造函数会在创建lock_guard对象时自动上锁互斥锁_Mtx。它使用互斥锁的lock()成员函数进行上锁操作。
  2. lock_guard(_Mutex& _Mtx, adopt_lock_t):该构造函数用于在已经上锁的情况下创建lock_guard对象。在这种情况下,不需要再次上锁互斥锁,因此该构造函数空实现。
  3. ~lock_guard() _NOEXCEPT:析构函数会在lock_guard对象销毁时自动解锁互斥锁_Mtx。它使用互斥锁的unlock()成员函数进行解锁操作。
  4. lock_guard(const lock_guard&) = deletelock_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更加的灵活,提供了更多的成员函数

  1. 上锁/解锁操作locktry_locktry_lock_fortry_lock_untilunlock
  2. 修改操作:移动赋值、交换(swap():与另一个unique_lock对象互换所管理的互斥量所有权)、释放(release():返回它所管理的互斥量对象的指针,并释放所有权)
  3. 获取属性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_lockmtx 进行上锁操作。在 main 函数中,我们创建了两个线程分别执行 work_in_critical_section 函数。由于使用了 unique_lock,它会在离开作用域时自动解锁互斥量,确保线程安全。

四、总结

RAII机制是一种重要的编程范式,Lock_guardUnique_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标准中&#xff0c;为了更方便地使用互斥锁&#xff08;Mutex&#xff09;来保…...

电路设计(8)——计时器的multism仿真

1.功能设计 这是一个计时电路&#xff0c;在秒脉冲的驱动下&#xff0c;计时器开始累加&#xff0c;6个数码管分别显示计时的 时&#xff1a;分&#xff1a;秒。 仿真图如下所示&#xff1a; 左边的运放构成了振荡电路&#xff0c;可以产生脉冲波。这个脉冲波给计时电路提供基准…...

Jmeter测试实践:文件下载接口

一 Jmeter步骤 1.打开jmeter4.0&#xff0c;新建测试计划&#xff0c;添加线程组。根据实际情况配置线程属性。 2.添加HTTP请求。根据接口文档进行配置。 Basic部分修改如下&#xff0c;Advanced部分保持默认。这里的参数id是文件的id&#xff0c;我进行了参数化&#xff0c…...

PyQt5实现学生管理系统第三天(下)

目录 一:学生课程导航 二:搜索框 三:查询 四:页面数据展示逻辑 上一节,我们介绍了课程管理的课程查询导航的功能。这一节我们介绍下学生课程的功能实现,因为学生课程只有一个查询列表,内容相对简单,所以我们在这一节也重点讲述下我们页面的展现逻辑。 一:学生课程…...

第4章 | 安徽某高校《统计建模与R软件》期末复习

第4章 参数估计 参数估计是统计建模的关键步骤之一&#xff0c;它涉及根据样本数据推断总体参数的过程。在统计学中&#xff0c;参数通常用于描述总体的特征&#xff0c;如均值、方差等。通过参数估计&#xff0c;我们可以利用样本信息对这些未知参数进行推断&#xff0c;从而…...

localforage本地存储(融合Web Storage,Web SQL Database,ndexedDB三种前端存储)

介绍 localForage 是一个快速而简单的 JavaScript 存储库。通过使用异步存储&#xff08;IndexedDB 或 WebSQL&#xff09;和简单的类 localStorage 的 API &#xff0c;localForage 能改善 Web 应用的离线体验。 在不支持 IndexedDB 或 WebSQL 的浏览器中&#xff0c;localF…...

【JavaWeb学习笔记】17 - ThreadLocal

项目代码 https://github.com/yinhai1114/JavaWeb_LearningCode/tree/main/threadlocal/src/com/yinhai/thread 目录 项目代码 一、什么是ThreadLocal? 二、ThreadLocal快速入门 三、源码解读 一、什么是ThreadLocal? 1. ThreadLocal的作用&#xff0c;可以实现在同一个线…...

【ARMv8M Cortex-M33 系列 1 -- SAU 介绍】

文章目录 Cortex-M33 SAU 介绍SAU 的主要功能包括SAU 寄存器配置示例 Cortex-M33 SAU 介绍 在 ARMv8-M 架构中&#xff0c;SAU&#xff08;Security Attribution Unit&#xff09;是安全属性单元&#xff0c;用于配置和管理内存区域的安全属性。SAU 是 ARM TrustZone 技术的一…...

sklearn 逻辑回归Demo

逻辑回归案例 假设表示 基于上述情况&#xff0c;要使分类器的输出在[0,1]之间&#xff0c;可以采用假设表示的方法。 设 h θ ( x ) g ( θ T x ) h_θ (x)g(θ^T x) hθ​(x)g(θTx)&#xff0c; 其中 g ( z ) 1 ( 1 e − z ) g(z)\frac{1}{(1e^{−z} )} g(z)(1e−z)1​…...

什么是众创空间?他有什么特点?

众创空间&#xff0c;是一种为大众创新创业提供专业化服务的创业服务平台&#xff0c;是顺应网络时代创新创业特点和需求&#xff0c;通过市场化机制、专业化服务和资本化途径构建的低成本、便利化、全要素、开放式的新型创业服务平台的统称。众创空间包括创客空间、联合办公空…...

什么是数据分析思维

参考 一文学会如何做电商数据分析&#xff08;附运营分析指标框架&#xff09; 电子商务该如何做数据分析&#xff1f;如何数据分析入门&#xff08;从各项指标表象进入&#xff09; https://www.processon.com/outline/6589838c3129f1550cc69950 数据分析步骤 什么是数据分析…...

利用Milvus Cloud和LangChain构建机器人:一种引人入胜且通俗易懂的方法

一、引言 机器人已经深入我们的日常生活&#xff0c;从家庭服务到工业生产&#xff0c;再到医疗和运输等领域。然而&#xff0c;这些机器人往往需要复杂的算法和数据处理技术才能有效地执行任务。在这个过程中&#xff0c;人工智能&#xff08;AI&#xff09;和机器学习&#…...

数据结构-如何实现一个队列?逐步解析与代码示例(超详细)

文章目录 前言1.队列的基本概念2.链表与数组实现队列的区别2.1数据存储结构2.2性能2.3内存使用 3.为什么选择链表实现队列&#xff1f;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以后&#xff0c;System.Drawing.Common开关已经被删除&#xff0c;且System.Drawing.Common仅在 Windows 上支持 &#xff0c;于是想办法将原来上传图片验证文件名和获取图片扩展名方法替换一下&#xff0c;便开始搜索相关解决方案。 .Net6.0文档&#xff1a;…...

【MySQL】InnoDB和MyISAM区别

文章目录 一、索引不同1 InnoDB聚簇索引&#xff0c;MyISAM非聚簇索引1 InnoDB聚簇索引2 MyISAM非聚簇索引 2 InnoDB必须要有主键&#xff0c;MyISAM允许没有主键3 InnoDB支持外键4 InnoDB不支持全文索引5 索引保存位置不同 二、对事物的支持三、存储结构不同四、存储空间不同五…...

3分钟了解安全数据交换系统有什么用!

企业为了保护核心数据安全&#xff0c;都会采取一些措施&#xff0c;比如做网络隔离划分&#xff0c;分成了不同的安全级别网络&#xff0c;或者安全域&#xff0c;接下来就是需要建设跨网络、跨安全域的安全数据交换系统&#xff0c;将安全保障与数据交换功能有机整合在一起&a…...

记录汇川:MODBUS TCP-梯形图

H5U的MODBUS通信不需要编写程序&#xff0c;通过组态MODBUS通信配置表&#xff0c;实现数据通信。 Modbus-TCP 主站即Modbus-TCP客户端&#xff0c;通过Modbus-TCP配置&#xff0c;可最多支持同时与31个 Modbus-TCP服务器&#xff08;从站&#xff09;进行通讯。 …...

electron + sqlite3 解决打包后无法写入数据库

前言 window环境。 electron28.0.0 sqlite35.1.6 使用 electron-builder 打包。 本文旨在解决打包后无法写入数据库的问题。 但如果你是打包后无法访问sqlite&#xff0c;且有报错弹窗&#xff0c;不妨也看看本文。 也许是同一种原因。 错误原因分析 打包后无法创建db文件&…...

【uniapp小程序-生成二维码+多个图片文字合并一张图】

<!-- 二维码 --><canvas id"qrcode" canvas-id"qrcode" width"120" ></canvas><!-- 生成带小程序码的分享图片 --><canvas canvas-id"shareCanvas" class"share-canvas"></canvas>#qrc…...

利用ngx_stream_return_module构建简易 TCP/UDP 响应网关

一、模块概述 ngx_stream_return_module 提供了一个极简的指令&#xff1a; return <value>;在收到客户端连接后&#xff0c;立即将 <value> 写回并关闭连接。<value> 支持内嵌文本和内置变量&#xff08;如 $time_iso8601、$remote_addr 等&#xff09;&a…...

PHP和Node.js哪个更爽?

先说结论&#xff0c;rust完胜。 php&#xff1a;laravel&#xff0c;swoole&#xff0c;webman&#xff0c;最开始在苏宁的时候写了几年php&#xff0c;当时觉得php真的是世界上最好的语言&#xff0c;因为当初活在舒适圈里&#xff0c;不愿意跳出来&#xff0c;就好比当初活在…...

java调用dll出现unsatisfiedLinkError以及JNA和JNI的区别

UnsatisfiedLinkError 在对接硬件设备中&#xff0c;我们会遇到使用 java 调用 dll文件 的情况&#xff0c;此时大概率出现UnsatisfiedLinkError链接错误&#xff0c;原因可能有如下几种 类名错误包名错误方法名参数错误使用 JNI 协议调用&#xff0c;结果 dll 未实现 JNI 协…...

关于iview组件中使用 table , 绑定序号分页后序号从1开始的解决方案

问题描述&#xff1a;iview使用table 中type: "index",分页之后 &#xff0c;索引还是从1开始&#xff0c;试过绑定后台返回数据的id, 这种方法可行&#xff0c;就是后台返回数据的每个页面id都不完全是按照从1开始的升序&#xff0c;因此百度了下&#xff0c;找到了…...

1688商品列表API与其他数据源的对接思路

将1688商品列表API与其他数据源对接时&#xff0c;需结合业务场景设计数据流转链路&#xff0c;重点关注数据格式兼容性、接口调用频率控制及数据一致性维护。以下是具体对接思路及关键技术点&#xff1a; 一、核心对接场景与目标 商品数据同步 场景&#xff1a;将1688商品信息…...

Cinnamon修改面板小工具图标

Cinnamon开始菜单-CSDN博客 设置模块都是做好的&#xff0c;比GNOME简单得多&#xff01; 在 applet.js 里增加 const Settings imports.ui.settings;this.settings new Settings.AppletSettings(this, HTYMenusonichy, instance_id); this.settings.bind(menu-icon, menu…...

微服务商城-商品微服务

数据表 CREATE TABLE product (id bigint(20) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT 商品id,cateid smallint(6) UNSIGNED NOT NULL DEFAULT 0 COMMENT 类别Id,name varchar(100) NOT NULL DEFAULT COMMENT 商品名称,subtitle varchar(200) NOT NULL DEFAULT COMMENT 商…...

佰力博科技与您探讨热释电测量的几种方法

热释电的测量主要涉及热释电系数的测定&#xff0c;这是表征热释电材料性能的重要参数。热释电系数的测量方法主要包括静态法、动态法和积分电荷法。其中&#xff0c;积分电荷法最为常用&#xff0c;其原理是通过测量在电容器上积累的热释电电荷&#xff0c;从而确定热释电系数…...

stm32wle5 lpuart DMA数据不接收

配置波特率9600时&#xff0c;需要使用外部低速晶振...

五、jmeter脚本参数化

目录 1、脚本参数化 1.1 用户定义的变量 1.1.1 添加及引用方式 1.1.2 测试得出用户定义变量的特点 1.2 用户参数 1.2.1 概念 1.2.2 位置不同效果不同 1.2.3、用户参数的勾选框 - 每次迭代更新一次 总结用户定义的变量、用户参数 1.3 csv数据文件参数化 1、脚本参数化 …...