目录

1.unique_lock取代lock_guard

2. unique_lock的第二个参数

2.1 std::adopt_lock

2.2 std::try_to_lock

2.3 std::defer_lock

3. unique_lock的成员函数

3.1 lock(),加锁

3.2 unlock(),解锁

3.3 try_lock()

3.4 release()

4. unique_lock所有权的传递


1.unique_lock取代lock_guard

unique_lock是个类模板,工作中,一般lock_guard(推荐使用);lock_guard取代了mutex的lock()和unlock();

unique_lock比lock_guard灵活很多,效率上差一点,内存占用多一点。

2. unique_lock的第二个参数

lock_guard可以带第二个参数:

std::lock_guard<std::mutex> sbguard1(my_mutex1, std::adopt_lock);// std::adopt_lock标记作用;

2.1 std::adopt_lock

表示这个互斥量已经被lock了(你必须要把互斥量提前lock了 ,否者会报异常);

std::adopt_lock标记的效果就是假设调用一方已经拥有了互斥量的所有权(已经lock成功了);通知lock_guard不需要再构造函数中lock这个互斥量了。

unique_lock也可以带std::adopt_lock标记,含义相同,就是不希望再unique_lock()的构造函数中lock这个mutex。

用std::adopt_lock的前提是,自己需要先把mutex lock上;用法与lock_guard相同。

2.2 std::try_to_lock

我们会尝试用mutex的lock()去锁定这个mutex,但如果没有锁定成功,我也会立即返回,并不会阻塞在那里;

用这个try_to_lock的前提是你自己不能先lock。实例代码如下:

#include<iostream>
#include<thread>
#include<string>
#include<vector>
#include<list>
#include<mutex>using namespace std;class A
{
public:void inMsgRecvQueue(){for (int i = 0; i < 10000; i++){cout << "inMsgRecvQueue()执行,插入一个元素" << i << endl;{ std::unique_lock<std::mutex> sbguard(my_mutex, std::try_to_lock);if (sbguard.owns_lock()){//拿到了锁msgRecvQueue.push_back(i); //...//其他处理代码}else{//没拿到锁cout << "inMsgRecvQueue()执行,但没拿到锁头,只能干点别的事" << i << endl;}}}}bool outMsgLULProc(int &command){my_mutex.lock();//要先lock(),后续才能用unique_lock的std::adopt_lock参数std::unique_lock<std::mutex> sbguard(my_mutex, std::adopt_lock);std::chrono::milliseconds dura(20000);std::this_thread::sleep_for(dura);  //休息20sif (!msgRecvQueue.empty()){//消息不为空int command = msgRecvQueue.front();//返回第一个元素,但不检查元素是否存在msgRecvQueue.pop_front();//移除第一个元素。但不返回;return true;}return false;}//把数据从消息队列取出的线程void outMsgRecvQueue(){int command = 0;for (int i = 0; i < 10000; i++){bool result = outMsgLULProc(command);if (result == true){cout << "outMsgRecvQueue()执行,取出一个元素" << endl;//处理数据}else{//消息队列为空cout << "inMsgRecvQueue()执行,但目前消息队列中为空!" << i << endl;}}cout << "end!" << endl;}private:std::list<int> msgRecvQueue;//容器(消息队列),代表玩家发送过来的命令。std::mutex my_mutex;//创建一个互斥量(一把锁)
};int main()
{A myobja;std::thread myOutMsgObj(&A::outMsgRecvQueue, &myobja);std::thread myInMsgObj(&A::inMsgRecvQueue, &myobja);myOutMsgObj.join();myInMsgObj.join();cout << "主线程执行!" << endl;return 0;
}

2.3 std::defer_lock

用std::defer_lock的前提是,你不能自己先lock,否则会报异常

std::defer_lock的意思就是并没有给mutex加锁:初始化了一个没有加锁的mutex。

我们借着defer_lock的话题,来介绍一些unique_lock的重要成员函数

3. unique_lock的成员函数

3.1 lock(),加锁

3.2 unlock(),解锁

defer_lock、lock()与unlock() 实例代码 如下 :

void inMsgRecvQueue(){for (int i = 0; i < 10000; i++){cout << "inMsgRecvQueue()执行,插入一个元素" << i << endl;std::unique_lock<std::mutex> sbguard(my_mutex, std::defer_lock);//没有加锁的my_mutexsbguard.lock();//咱们不用自己unlock//处理共享代码//因为有一些非共享代码要处理sbguard.unlock();//处理非共享代码要处理。。。sbguard.lock();//处理共享代码msgRecvQueue.push_back(i);//...//其他处理代码sbguard.unlock();//画蛇添足,但也可以}}

3.3 try_lock()

尝试给互斥量加锁,如果拿不到锁,返回false,如果拿到了锁,返回true,这个函数是不阻塞的;实例代码如下:

void inMsgRecvQueue(){for (int i = 0; i < 10000; i++){std::unique_lock<std::mutex> sbguard(my_mutex, std::defer_lock);//没有加锁的my_mutexif (sbguard.try_lock() == true)//返回true表示拿到锁了{msgRecvQueue.push_back(i);//...//其他处理代码}else{//没拿到锁cout << "inMsgRecvQueue()执行,但没拿到锁头,只能干点别的事" << i << endl;}}}

3.4 release()

返回它所管理的mutex对象指针,并释放所有权;也就是说,这个unique_lock和mutex不再有关系。严格区分unlock()与release()的区别,不要混淆。

如果原来mutex对像处于加锁状态,你有责任接管过来并负责解锁。(release返回的是原始mutex的指针)。实例代码如下:

void inMsgRecvQueue(){for (int i = 0; i < 10000; i++){std::unique_lock<std::mutex> sbguard(my_mutex);std::mutex *ptx = sbguard.release(); //现在你有责任自己解锁了msgRecvQueue.push_back(i);ptx->unlock(); //自己负责mutex的unlock了}}

为什么有时候需要unlock();因为你lock()锁住的代码段越少,执行越快,整个程序运行效率越高。有人也把锁头锁住的代码多少成为锁的粒度,粒度一般用粗细来描述;

a)锁住的代码少,这个粒度叫细,执行效率高;

b)锁住的代码多,这个粒度叫粗,执行效率低;

要学会尽量选择合适粒度的代码进行保护,粒度太细,可能漏掉共享数据的保护,粒度太粗,影响效率。

选择合适的粒度是高级程序员能力和实力的体现;

4. unique_lock所有权的传递

std::unique_lock<std::mutex> sbguard(my_mutex);//所有权概念

sbguard拥有my_mutex的所有权;sbguard可以把自己对mutex(my_mutex)的所有权转移给其他的unique_lock对象;

所以unique_lock对象这个mutex的所有权是可以转移,但是不能复制。

std::unique_lock<std::mutex> sbguard1(my_mutex);

std::unique_lock<std::mutex> sbguard2(sbguard1);//此句是非法的,复制所有权是非法的

std::unique_lock<std::mutex> sbguard2(std::move(sbguard));//移动语义,现在先当与sbguard2与my_mutex绑定到一起了//现在sbguard1指向空,sbguard2指向了my_mutex

方法1 :std::move()

方法2:return std:: unique_lock<std::mutex>  代码如下:

std::unique_lock<std::mutex> rtn_unique_lock(){std::unique_lock<std::mutex> tmpguard(my_mutex);return tmpguard;//从函数中返回一个局部的unique_lock对象是可以的。三章十四节讲解过移动构造函数。//返回这种举报对象tmpguard会导致系统生成临时unique_lock对象,并调用unique_lock的移动构造函数}void inMsgRecvQueue(){for (int i = 0; i < 10000; i++){std::unique_lock<std::mutex> sbguard1 = rtn_unique_lock();msgRecvQueue.push_back(i);}}

注:该文是C++11并发多线程视频教程笔记,详情学习:https://study.163.com/course/courseMain.htm?courseId=1006067356

C++11多线程 unique_lock详解相关推荐

  1. IOS 多线程04-GCD详解 底层并发 API

    IOS 多线程04-GCD详解 底层并发 API 注:本人是翻译过来,并且加上本人的一点见解. 前言 想要揭示出表面之下深层次的一些可利用的方面.这些底层的 API 提供了大量的灵活性,随之而来的是大 ...

  2. 多线程——synchronized详解

    多线程--synchronized详解 "当多个线程同时访问一个对象时,如果不用考虑这些线程在运行时环境下 的调度和交替执行,也不需要进行额外的同步,或者在调用方进行任何其他的协调操作,调用 ...

  3. java线程的cancel_多线程-Cancel详解

    多线程-Cancel详解 在取消一个对等线程的请求被同意时,会有一个取消过程同pthread_cancel( )的返回异步发生. 目标线程的取消类型和取消状态决定了取消何时真正发生.可取消性状态描述了 ...

  4. mysql int(3)与int(11)的区别详解

    这篇文章主要介绍了mysql int(3)与int(11)的区别详解的相关资料,需要的朋友可以参考下 mysql int(3)与int(11)的区别 总结,int(M) zerofill,加上zero ...

  5. mysql8.0.11 安装顺序_mysql 8.0.11 安装步骤详解

    本文为大家分享了mysql 8.0.11 安装步骤,供大家参考,具体内容如下 第一步:下载安装包 MYSQL官方下载地址:官方下载 这里第一项是在线安装,第二项是离线包安装,我选择的是第二项(不用管你 ...

  6. java多线程设计模式详解[推荐]

    java多线程设计模式详解[推荐] java多线程设计模式详解之一 线程的创建和启动 java语言已经内置了多线程支持,所有实现Runnable接口的类都可被启动一个新线程,新线程会执行该实例的run ...

  7. [原创]手把手教你Linux下的多线程设计--Linux下多线程编程详解(一)

    本文可任意转载,但必须注明作者和出处. [原创]手把手教你Linux下的多线程设计(一)                                       --Linux下多线程编程详解 原 ...

  8. 多线程锁详解之【临界区】

    更多的锁介绍可以先看看这篇文章:多线程锁详解之[序章] 正文: 一般锁的类型可分为两种:用户态锁和内核态锁.用户态锁是指这个锁的不能够跨进程使用.而内核态锁就是指能够跨进程使用的锁.一般书中会说,wi ...

  9. WindowsMediaPlayer 11 控件详解

    WindowsMediaPlayer 11 控件详解 转载别人的VB内容 C#中大多也通用 都是Microsoft的作品 . 属性/方法名: 说明: [基本属性] URL:String; 指定媒体位置 ...

  10. Java多线程进阶详解

    文章目录 1.卖票案例引入数据不安全问题 2.同步代码块 深入理解synchronized关键字 3.同步方法与静态同步方法 同步方法 静态同步方法 内置锁 静态同步方法与同步代码块共同使用 为什么要 ...

最新文章

  1. redis sentinel集群配置及haproxy配置
  2. 网站关键词如何布局更有利于关键词排名提升?
  3. 这公司需要再利用CPU热能:于是淘汰i7换AMD Ryzen
  4. ArrayList刷题总结
  5. Dubbo-HelloWorld
  6. 计算机控制分离性原理是什么,(第12讲)状态观测器和分离原理.ppt
  7. Windows IP测试小脚本
  8. mysql自主增长键_mysql中关于自增长主键的获取
  9. AttributeError: module 'tensorflow' has no attribute 'python'
  10. Python---试除法求质数的三种方式对比
  11. SQL Server2016安装教程
  12. 软件工程-UML画图
  13. 优秀的WMS仓库管理系统应该具备哪些条件
  14. 内容太干准备好水:项目进度管理S曲线制作方法新探
  15. ubantu apt命令失败
  16. 数字化转型转的是什么
  17. JAVA入门——lesson 7
  18. 黑科技新添成员, 小米mix5再次创新, 但这些真的只是黑科技的全部吗?
  19. 中国电子学会2022年python六级考试真题大题 类与对象练习题
  20. Visual Studio 2019编译问题解决方法

热门文章

  1. Hadoop报错 Failed to locate the winutils binary in the hadoop
  2. [Bzoj2049][Sdoi2008]Cave 洞穴勘测
  3. javascript 网页自动跳转
  4. javascript Date定义和体验
  5. [日常工作]非Windows Server 系统远程经常断以及提高性能的方法
  6. Exchange管理界面
  7. [USACO2006][poj3182]The Grove(巧妙的BFS)
  8. 暂别ACM,转移阵地
  9. bootstrap-table初始化配置
  10. TCP异常终止(RESET报文)