C++线程中的几类锁

article/2025/8/19 1:36:53

C++线程中的几类锁

  多线程中的锁主要有五类:互斥锁、条件锁、自旋锁、读写锁、递归锁。一般而言,所得功能与性能成反比。而且我们一般不使用递归锁(C++提供std::recursive_mutex),这里不做介绍。

互斥锁

  互斥锁用于控制多个线程对它们之间共享资源互斥访问的一个信号量也就是说为了避免多个线程在某一时刻同时操作一个共享资源,例如一个全局变量,任何一个线程都要使用初始锁互斥地访问,以避免多个线程同时访问发生错乱。

  在某一时刻只有一个线程可以获得互斥锁,在释放互斥锁之前其它线程都不能获得互斥锁,以阻塞的状态在一个等待队列中等待。

头文件:#include <mutex>

类型:std::std::mutex、std::lock_guard

用法:在C++中,通过构造std::mutex的实例创建互斥单元,调用成员函数lock()来锁定共享资源,调用unlock()来解锁。不过一般不使用这种解决方案,更多的是使用C++标准库中的std::lock_guard类模板,实现了一个互斥量包装程序,提供了一种方便的RAII风格的机制在作用域块中。

关于RAII惯用法的介绍可参考博主的博客:https://blog.csdn.net/qq135595696/article/details/121411724。

示例代码:

#include <iostream>
#include <thread>//C++11线程库是跨平台的
#include <mutex>//C++互斥锁
#include <vector>
#include <windows.h>int g_num = 0;
std::mutex g_mutex;void ThreadFunc(int a)
{cout << "启动线程:" << a << endl;for (int i = 0; i < 1000000; i++){//g_mutex.lock();std::lock_guard<std::mutex> m(g_mutex);//互斥量包装程序g_num++;//g_mutex.unlock();}
}int main()
{for (int i = 0; i < 4; i++){std::thread t(ThreadFunc, i);t.detach();}Sleep(2000);cout << "g_num:" << g_num << endl;return 0;
}//高阶版,将上述main()函数的函数名更改,再更改以下的mainTest()即可执行。两个方法的执行的结果相同,原理也相同。
int mainTest()
{std::vector<std::thread *> ts;for (int i = 0; i < 4; i++){std::thread *t = new std::thread(ThreadFunc, i);//t.detach();ts.push_back(t);}for (auto begin = ts.begin(); begin != ts.end(); begin++)(*begin)->join();Sleep(2000);cout << "g_num:" << g_num << endl;return 0;
}

效果图

TIPS:注意std::cout和std::end都是线程不安全的,所以才会出现线程1和线程3在一行,原因就是线程1未执行cout<<endl。CPU的时间片就已经用完了,CPU转移执行线程3后,再执行线程1的cout<<endl。

具体C++11中thread库join和detach的区别可参考:https://blog.csdn.net/qq135595696/article/details/121387692

条件锁

  条件锁就是所谓的条件变量,当某一个线程因为某个条件未满足时可以使用条件变量使该程序处于阻塞状态,一旦条件满足则以“信号量”的方式唤醒一个因为该条件而被阻塞的线程。最为常见的就是在线程池中,初始情况下因为没有任务使得任务队列为空,此时线程池中的线程因为“任务队列为空”这个条件处于阻塞状态。一旦有任务进来,就会以信号量的方式唤醒该线程来处理这个任务。

自旋锁

  互斥锁和条件锁都是比较常见的锁,比较容易理解。接下来用互斥锁和自旋锁的原理相互比较,来理解自旋锁。

  假设我们有一台计算机,该计算机拥有两个处理器core1和core2.现在在这台计算机上运行两个线程:T1和T2,且T1和T2分别在处理器core1和core2上面运行,两个线程之间共享一份公共资源Public。

  首先我们说明互斥锁的工作原理,互斥锁是一种sleep-waiting的锁。假设线程T1访问公共资源Public并获得互斥锁,同时在core1处理器上运行,此时线程T2也想要访问这份公共资源Public(即想要获得互斥锁),但是由于T1正在使用Public使得T2被阻塞。当T2处于阻塞状态时,T2被放入等待队列中,处理器core2会去处理其它的任务而不必一直等待(忙等)。也就是说处理器不会因为线程被阻塞而空闲,它会去处理其它事务。

  然后我们说明自旋锁的工作原理,自旋锁是一种busy-waiting的锁。也就是说,如果T1正在使用Public,而T2也想使用Public,此时T2肯定是得不到这个自旋锁的。与互斥锁相反,此时运行T2的处理器core2会一直不断地循环检查Public使用可用(自旋锁请求),直到获得到这个自旋锁为止。

  从“自旋锁”的名称也可以看出,如果一个线程想要获得一个被使用的自旋锁,那么它会一直占用CPU请求这个自旋锁使得CPU不能去做其它的事情,知道获取这个锁为止,这就是“自旋”的含义。当发生阻塞时,互斥锁可以让CPU去处理其它的事务,但自旋锁让CPU一直不断循环请求获取这个锁。通过比较,我们可以明显的得出结论:“自旋锁”是比较消耗CPU的。

读写锁

  读写锁我们可以借助于“读者-写者”问题进行理解。接下来我们简单说下“读者-写者”问题。

  计算机中某些数据被多个进程共享,对数据库的操作有两种:一种是读操作,就是从数据库中读取数据不会修改数据库中内容;另一种就是写操作,写操作会修改数据库中存放的数据。因此可以得到我们允许在数据库上同时执行多个“读”操作,但是某一时刻只能在数据库上有一个“写”操作来更新数据。这就是简单的读者-写者模型。

参考博客

https://blog.csdn.net/xy_cpp/article/details/81910513


http://chatgpt.dhexx.cn/article/GUtxpGUG.shtml

相关文章

Oracle - 锁

锁概念 锁出现在数据共享的环境中&#xff0c;它是一种机制&#xff0c;在访问相同资源时&#xff0c;可以防止事务之间的破坏性交互。例如&#xff0c;在多个会话同时操作某表时&#xff0c;优先操作的会话需要对其锁定。 事务的分离性要求当前事务不能影响其他的事务&#…

用友数据库错误“未能读取并闩锁页(1:3355)(用闩锁类型SH)”修复

客户硬盘无法识别&#xff0c;检测后&#xff0c;硬盘有坏道&#xff0c;由于数据库正在坏道上&#xff0c;所以恢复出来的用友数据库无法附加。 通过无日志附加后&#xff0c;做DBCC检测数据库出现以下错误&#xff1a; “消息8966&#xff0c;级别16&#xff0c;状态1&#x…

Mysql中锁的类型有哪些?

Mysql中锁的类型有哪些&#xff1f; 1. 基于锁的属性分类&#xff1a;共享锁、排他锁2. 基于锁的粒度分类&#xff1a;行级锁&#xff08;INNODB&#xff09;、表级锁&#xff08;INNODB、MYISAM&#xff09;、页级锁&#xff08;BDB引擎&#xff09;、记录锁、间隙锁、临键锁。…

mysql 常见锁的类型(一)

文章目录 一、锁的分类1.1 加锁的目的1.2 锁的类别 二、乐观锁和悲观锁2.1. 乐观锁2.2. 悲观锁&#xff1a; 三、共享锁与排他锁四、表锁五、意向锁六、行级锁七、记录锁&#xff08;Record Locks&#xff09;八、间隙锁&#xff08;Gap Locks&#xff09;九、临键锁&#xff0…

MySQL-InnoDB常用锁类型解析

Shared&#xff08;乐观锁&#xff09; and Exclusive Locks&#xff08;互斥锁&#xff09;&#xff1a; InnoDB有两种锁类型&#xff0c;Shared&#xff08;s&#xff09; and Exclusive&#xff08;x&#xff09; Locks&#xff08;乐观锁和互斥锁&#xff09;。 Shared&…

MySql InnoDB锁类型

MySql InnoDB锁类型 从类型上来分类&#xff0c;InnoDB存储引擎实现了两种标准的锁 共享锁(S-Lock)&#xff1a;允许事务读一行数据 排它锁(X-Lock)&#xff1a;允许事务删除或者更新一行数据 如果一个事务获取了S锁&#xff0c;那么其他事务也可以立即获得S锁&#xff0c;…

锁的分类总结

锁的分类是从不同角度去看的。同一个锁也可以同时属于多种类型。 一、乐观锁与悲观锁 1. 互斥同步锁的劣势 阻塞和唤醒会带来性能的劣势 用户态和核心态切换上下文切换检查是否有被阻塞线程需要被唤醒等等 可能出现永久阻塞的问题&#xff1a;持有锁的线程永久阻塞了&#…

锁的介绍和分类(轻量级锁 重量级锁 偏向锁 自旋锁 互斥锁)

目录 公平锁 非公平锁 非公平锁 公平锁 可重入锁 不可重入锁 可重入锁&#xff08;递归锁&#xff09; 不可重入锁 轻量级锁 重量级锁 偏向锁 重量级锁 自旋锁(循环上锁) 轻量级锁 轻量级锁的释放 偏向锁 自旋锁和互斥锁 自旋锁 互斥锁 为何要使用自旋锁 自旋…

最全锁种类

你可能听说过很多锁&#xff0c;也看到过很多文章讲解锁&#xff0c;这篇我在这里将对锁的不同分类进行描述锁的设计 互斥锁–共享锁 互斥锁&#xff1a;顾名思义&#xff0c;就是互斥的&#xff0c;意思就是当前同步代码块只能被一个线程访问&#xff0c;sync、reentrantlock、…

锁的类型有哪些

锁的类型有哪些 基于锁的属性分类&#xff1a;共享锁、排他锁。 基于锁的粒度分类&#xff1a;行级锁(INNODB)、表级锁(INNODB、MYISAM)、页级锁(BDB引擎 )、记录锁、间隙锁、临键锁。 基于锁的状态分类&#xff1a;意向共享锁、意向排它锁 共享锁(Share Lock) 共享锁又称读锁&…

Kettle使用教程之Job使用

1、Kettle的Job使用十分简单&#xff0c;这里也只是演示比较简单的操作&#xff0c;创建Job 2、点击转换&#xff0c;然后点击浏览&#xff0c;选择转换对象 3、执行按钮&#xff0c;运行该转换 4、如果需要长期的进行定时转换&#xff0c;可以在Job中的start控件进行配置 转载…

Kettle使用教程之数据同步

Kettle使用教程之数据同步 数据模型原型如下&#xff1a; 1、表输入&#xff0c;针对最新的数据输入的表 2、目标表&#xff0c;需要更新的表 3、两个表都需要进行排序操作 4、合并&#xff0c;根据id进行合并 5、数据同步(包括更新、插入、删除) 6、点击运行&#xff0c;就可…

ETL开发工具KETTLE使用教程

Kettle的建立数据库连接、使用kettle进行简单的全量对比插入更新&#xff1a;kettle会自动对比用户设置的对比字段&#xff0c;若目标表不存在该字段&#xff0c;则新插入该条记录。若存在&#xff0c;则更新。 Kettle简介&#xff1a;Kettle是一款国外开源的ETL工具&#xff0…

ETL工具Kettle使用教程

Kettle使用教程之数据同步 数据同步标识字段 标志字段的值有4种&#xff0c;分别是&#xff1a; “Identical” : 关键字段在新旧数据源中都存在&#xff0c;且域值相同 “changed” : 关键字段在新旧数据源中都存在&#xff0c;但域值不同 “new” : 旧数据源中没有找到该…

Kettle使用教程(一)—— 在MacOS系统中安装 Kettle

Kettle使用教程&#xff08;一&#xff09;—— 在MacOS系统中安装 Kettle 一、环境准备二、下载并启动Kettle二、初始化资源库 一、环境准备 Kettle 9.2JDK 1.8 &#xff08;安装指引&#xff09;Mysql&#xff08;安装指引&#xff09; 二、下载并启动Kettle 首先到官网下…

Kettle使用教程(问题)

关于kettle的介绍此文不做介绍 笔者电脑环境 winoraclejdk1.8kettle7.1 1. 考虑到在在官网下载速度比较慢&#xff0c;在这里可以使用国内的镜像 国内镜像 2. 配置java环境 (1) kettle需要以来java环境&#xff0c;因为没有安装java环境的朋友请移步配置java环境&#xff…

R-P-Faster R-CNN day65 读论文:高分辨率遥感影像综合地理空间目标检测框架

An Efficient and Robust Integrated Geospatial Object Detection Framework for High Spatial Resolution Remote Sensing Imagery 1. Introduction3. Overview of the Proposed R-P-Faster R-CNN Framework3.1. 有效集成区域建议网络与目标检测Faster R-CNN框架3.1.2. RPN与…

多特征融合的高分辨率遥感图像海陆分离——刘思彤

论文阅读笔记 摘要 为解决目前大多数海陆分离方法仅利用单一特征对图像进行处理&#xff0c;从而导致误分割或存在大量孤立区域&#xff0c;造成后续处理工作难度大或无法开展的问题&#xff0c;提出一种联合灰度、梯度和纹理等多特征的海陆分离方法。针对不同的陆地类型选用…

从实现的技术手段细数:高分辨率遥感影像+人工智能的现在和未来

【转自】http://baijiahao.baidu.com/s?id1601901487917871613&wfrspider&forpc 人工智能不可阻挡地向各行业渗透。这一现象&#xff0c;恰巧撞上了“商业遥感卫星发射潮”。这似乎预示了某种潜流。此刻&#xff0c;从事卫星遥感影像解译和大数据提取的专业人士、科研…

高分辨率光学遥感影像舰船目标检测与识别算法研究(尹莹莹)

论文阅读笔记 摘要 本文主要研究海陆背景下的光学遥感图像舰船目标检测与识别技术&#xff0c;重点研究了海陆分离、舰船目标疑似区域检测技术与疑似区域目标识别技术。 海陆分离&#xff1a; 采用了OTSU与形态学相结合的方法实现海路区域初步划分&#xff1b;再以孤立区域内…