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

lock_guard和unique_lock学习总结

1.std::lock_guard

std::lock_guard其实就是简单的RAII(Resource Acquisition Is Initialization)封装,资源获取即初始化。在构造函数中进行加锁,析构函数中进行解锁,这样可以保证函数退出时,锁一定被释放。 

不可以对 std::lock_guard 调用 unlock 进行解锁操作。std::lock_guard 是一个非常简单的互斥量管理类,设计初衷是提供一种 RAII(资源获取即初始化)风格的锁机制,当 std::lock_guard 对象创建时锁定互斥量,并在对象销毁时析构函数里自动解锁互斥量。std::lock_guard 没有提供解锁和重新锁定的功能。它的接口非常简单,只包含构造函数和析构函数,没有 unlock 或 lock 方法。这使得 std::lock_guard 更加轻量级,但也更不灵活。 

构造函数:

explicit lock_guard(mutex_type& m); // 引用类型,锁定互斥量m,析构函数中解锁// 其中mutex_type是模板参数的别名
template <class Mutex>
class lock_guard {
public:using mutex_type = Mutex; // 定义 mutex_type 为模板参数 Mutexexplicit lock_guard(mutex_type& m);~lock_guard();// ...
};

2.std::unique_lock

转自:https://www.cnblogs.com/moodlxs/p/10111843.html

std::lock_guard的功能超集, 封装了各种加锁操作,阻塞的,非阻塞的,还可以结合条件变量一起使用,基本上对锁的各种操作都封装了。所以性能和内存开销都比std::lock_guard大得多,需要有选择地使用。 std::unique_lock也会在析构的时候自动解锁,所以说,是std::lock_guard的功能超集。

支持多种加锁模式,构造函数:

unique_lock() noexcept; // 默认构造函数创建一个不与任何互斥量相关联的 unique_lock 对象
explicit unique_lock(mutex_type& m); //创建一个与互斥量 m 相关联并立即锁定它的 unique_lock 对象,通过析构函数unlock
unique_lock( mutex_type& m, std::defer_lock_t t );   //延迟加锁,之后再调用lock()主动加锁,可以不用unlock,可以通过unique_lock的析构函数unlock
unique_lock( mutex_type& m, std::try_to_lock_t t ); //尝试加锁
unique_lock( mutex_type& m, std::adopt_lock_t t );   //马上加锁
unique_lock(unique_lock&& u) noexcept; // 移动构造函数。原来的 unique_lock 对象将不再持有互斥量。

尝试加锁例子:

std::mutex mlock;void work1(int& s) {for (int i = 1; i <= 5000; i++) {std::unique_lock<std::mutex> munique(mlock, std::try_to_lock);if (munique.owns_lock() == true) {// 判断是否加锁成功s += i;}else {// 执行一些没有共享内存的代码}}
}

延迟加锁例子:

void work1(int& s) {for (int i = 1; i <= 5000; i++) {std::unique_lock<std::mutex> munique(mlock, std::defer_lock);if (munique.try_lock() == true) {// 判断当前是否能locks += i;}else {// 处理一些没有共享内存的代码}}
}

移动构造函数例子:

std::unique_lock<std::mutex> munique1(mlock);
std::unique_lock<std::mutex> munique2(std::move(munique1));
// 此时munique1失去mlock的权限,并指向空值,munique2获取mlock的权限

http://www.lryc.cn/news/453077.html

相关文章:

  • 数据挖掘-padans初步使用
  • 小阿轩yx-案例:项目发布基础
  • 【HarmonyOS】时间处理Dayjs
  • 论React Native 和 UniApp 的区别
  • 微信小程序处理交易投诉管理,支持多小程序
  • Pikachu-xss防范措施 - href输出 js输出
  • 数据结构双向链表和循环链表
  • go基础面试题汇总第一弹
  • Redis 实现分布式锁时需要考虑的问题
  • 百年极限论一直存在百年糊涂话:有正数小于所有正数
  • 红日靶场1学习笔记
  • 【C++篇】揭开 C++ STL list 容器的神秘面纱:从底层设计到高效应用的全景解析(附源码)
  • 【C#生态园】打造现代化跨平台应用:深度解析.NET桌面应用工具
  • 第二十一章 (动态内存管理)
  • 机器学习框架总结
  • docker pull 超时的问题如何解决
  • 【数学分析笔记】第4章第3节 导数四则运算和反函数求导法则(2)
  • 【2024】基于mysqldump的数据备份与恢复
  • 家用无线路由器配置
  • 模拟算法(4)_外观数列
  • vsomeip用到的socket
  • MFC有三个选项:MFC ActiveX控件、MFC应用程序、MFC DLL,如何选择?
  • 边缘概率 | 条件概率
  • 深入浅出:现代JavaScript开发者必知必会的Web性能优化技巧
  • 【S32K3 RTD LLD篇5】K344 ADC SW+HW trigger
  • TransFormer 视频笔记
  • 前端的混合全栈之路Meteor篇(三):发布订阅示例代码及如何将Meteor的响应数据映射到vue3的reactive系统
  • 自动驾驶系列—颠覆未来驾驶:深入解析自动驾驶线控转向系统技术
  • Webstorm 中对 Node.js 后端项目进行断点调试
  • VUE前后端分离毕业设计题目项目有哪些,VUE程序开发常见毕业论文设计推荐