C++11——线程库的理解与使用

目录

前言

一、线程库的构造

1.默认构造 

2.带参构造 

3.拷贝构造与赋值拷贝(不支持)

4.移动构造

二、线程调用lambda函数

三、线程安全与锁

1.lambda中的线程与锁

2.函数指针中的线程与锁

3.trylock()

4.recursive_mutex

5.lock_gurad守卫锁

6.unique_lock

四、条件变量

五、atomic

六、shared_ptr的多线程问题

七、懒汉模式中的线程安全问题


前言

之前我们学习过Linux的线程库thread,他是使用 POSIX 标准,而Windows中的线程库,使用的其他标准,在C++11之前他们的接口是不一样的,对代码的编写就很麻烦,Linux一套,Windows一套。C++11出来之后,引入了标准的线程库,使得在不同平台上编写具有可移植性的多线程代码变得更加容易。

一、线程库的构造

之前我们创建线程,都要先pthread_t 创建一个线程tid,然后把线程id,线程属性,一个函数指针,还有参数都传递过去。如果参数较多,我们还得封装成一个结构体,把结构体地址传过去。这并不是很方便,也不是很体面。

1.默认构造 

C++11提供了线程的默认构造(创建线程,但不启动)

 使用方法很简单,创建即可,一般要配合移动构造使用。(后面会讲)

2.带参构造 

带参构造,第一个参数为可调用对象(函数指针,仿函数,lambda函数,包装器)并且函数返回值并不限制,第二个参数为可变参数包,也就是我们不需要再有一个线程id,和设置线程属性,同时也不需 要再封装成一个结构体了,直接将参数传递给参数包,他自动解析就可以了。

现在我们来使用一下C++11的线程构造,如下,让 t 线程去执行Print函数。其中this_thread作用域下的get_id()能获取当前进程的id。

运行发现让id为19524的线程去执行了函数。 

多个参数也很简单,直接传递参数即可,不再需要使用结构体。 

3.拷贝构造与赋值拷贝(不支持)

thread库并支持拷贝构造与赋值拷贝,如果允许拷贝构造或拷贝赋值,那么会导致多个线程对象管理同一个底层线程,这可能会引发竞争条件和资源管理问题。

4.移动构造

线程的移动构造是在线程即将死亡(被回收)的时候,把资源给另外一个线程。因为这样线程依然只属于一个人,不会造成多个线程对象管理同一个底层线程的情况。

同时他能配合默认构造一起使用,因为默认构造,你根本都没传递函数,你怎么能让线程去处理任务呢?配合移动构造,就能让默认构造的线程也跑起来。

move是将t2转为将亡值。t1去夺舍t2。

同时,这样我们也可以使用容器管理线程了。因为反正有默认构造,vector只传入 int 整形就是在进行默认构造。后面再通过移动赋值让线程运行。

 小总结:

  1. 带参构造,创建可执行线程
  2. 先创建空线程对象,移动构造或者移动赋值将右值对象转移过去

二、线程调用lambda函数

我们知道,线程需要调用一个可调用对象,让他去执行这个可调用对象,从而让线程运行起来。其中运用最多的就是普通函数和lambda函数,这是因为让线程执行任务,我们只需要把任务说明白就行,包装器更适合提取函数的类型,仿函数又不够轻量化。

线程调用lambda函数比较简单,代码如下,在lambda捕获列表进行捕获就可以了。

三、线程安全与锁

1.lambda中的线程与锁

我们定义一个变量,让两个线程同时去对这个变量做++操作,按道理结果应该是20000,但是有可能结果不如我们的预期,这就是多线程导致的数据安全问题,++x并不是原子操作。

我们需要对临界资源进行上锁,来保证临界资源的安全。C++11的mutex库提供的mutex默认构造,直接使用即可

代码如下,对++x进行加锁与解锁 

2.函数指针中的线程与锁

在函数指针中,我们也传递一把锁,让他去保护临界资源x,这里都传递的是引用,按照之前的学习,我们的代码是没有问题的,这里却发生报错,编译不通过。

这是因为你传递的参数是传递给thread带参构造的可变参数包的,并不是我们看到的直接传参,他还会有一些处理。

thread 构造函数会对传递的参数进行拷贝或移动,然后将这些拷贝或移动后的参数传递给线程函数,导致在新线程中修改的实际是拷贝或移动后的引用,而不是原始的引用,因此会发生错误。

添加ref代表强制引用, 因此我们这里记住添加ref就好

这确实比较麻烦,我们也可以选择用指针,就可以避免这些问题。

3.trylock()

锁的lock()如果现在申请不到锁,就会在锁的等待队列上等待,直到轮到了自己,才会申请锁成功。而trylock()申请锁失败不会进入等待队列进行等待,而是返回false,继续往后执行。

4.recursive_mutex

revursive_mutex是递归互斥锁,线程在持有锁的情况下,再递归调用自己的函数,如果是普通的互斥锁就会发生死锁,此时需要使用递归互斥锁,防止死锁。

5.lock_gurad守卫锁

再我们对临界资源进行加锁解锁时,可能会发生一些意外,比如把解锁写错了,写成加锁

当然,这个错误比较低级,但如果临界区代码发生了异常呢?

大家看如下代码,我们在加锁与解锁中模拟了一个异常情况。

如果发生了异常,catch 块中的代码会被执行,然后程序会继续执行 try-catch 块之后的代码,而不会回到抛出异常的地方继续执行,于是我们的锁就不会解锁,后续线程想申请锁,就申请不到了,这会导致死锁的发生。

因此我们需要利用RAII的思想,使用lock_guard来守护线程,也就是把锁资源交给一个类,让类构造时申请资源,析构时自动释放资源。

使用如下代码进行mutex资源守护,注意成员变量和构造参数一定要引用,因为锁是不支持拷贝的,使用引用代表指的一直是这一把锁。 

template<class Lock>
class LockGuard
{
public:
	LockGuard(Lock& lk)
		:_lk(lk)
	{
		_lk.lock();
	}
	~LockGuard()
	{
		_lk.unlock();
	}
private:
	Lock& _lk;
};

那么线程使用上了LockGuard,出了作用域会自动析构,也就是说你catch捕获异常的时候,我就会析构了,然后释放锁资源,就不会死锁了。 

 std库也给我们设计好了 lock_guard,拿来用就可以了。

6.unique_lock

unique_lock也可以完成守卫锁的任务,他比守卫锁多了手动的加锁与解锁,同时可以与time_mutex进行配合。

四、条件变量

condition_variable就是条件变量,他只有默认构造。同时wait函数需要传递的锁是unique_lock类型。传入unique_lock就是要让wait先去解锁之后再去等待,lock_guard不支持手动解锁

条件变量本质就是通知,告诉等待你可以去再申请锁了。notify_one是通知某一个线程,notify_all通知所有线程。如果没有线程等待,就不做处理。

如下代码,就运用了条件变量,实现让线程1线程2轮流打印。

首先是使用unique_lock进行加锁,一开始flag为false,因此t1线程不会去等待,肯定是t1线程先打印,再将flag置为true,再通知t2线程。

此时t2线程要么在锁的地方阻塞住,要么就比t1线程更先运行,已经判断过flag为 false了,在wait中进行等待,被通知了就继续运行了,因此就可以实现交替打印了。

#include<iostream>
#include<thread>
#include<vector>
#include<mutex>
using namespace std;
int main()
{
	int n = 20;
	bool flag = false;
	mutex mtx;
	condition_variable cv;
	thread t1([n, &flag,&mtx,&cv] {
		for (int i = 1; i <= n; i++)
		{
			if(i%2==1)
			{
				unique_lock<mutex> lock(mtx);//加锁,出作用域自动解锁
				if (flag)
				{
					cv.wait(lock);//wait先解锁在去等待队列等待
				}
				cout << i << endl;
				flag = true;
				cv.notify_one();//信号变量通知其他线程取消等待
			}
		}
		});
	thread t2([n, &flag, &mtx, &cv] {
		for (int i = 1; i <= n; i++)
		{
			if (i % 2 == 0)
			{
				unique_lock<mutex> lock(mtx);//加锁,出作用域自动解锁
				if (!flag)
				{
					cv.wait(lock);//wait先解锁在去等待队列等待
				}
				cout << i << endl;
				flag = false;
				cv.notify_one();//信号变量通知其他线程取消等待
			}
		}
		});
	t1.join();
	t2.join();
}

五、atomic

前面我们的代码临界区都不算很长,此时使用互斥锁的效率就会变得很低,因为互斥锁保证原子操作,会导致线程切换时间片浪费的情况。

比如线程 thread_1 加锁了,正在++x,还没有解锁,此时时间片到了。被切换了,线程 thread_2 来了,想去申请锁,却一直申请不到,就会在等待队列等待,白白浪费了自己的时间片。C++11提供了atomic来保证变量的原子性。

当然,只建议对内置类型的处理,如果传入的类型是自定义类型,代码比较长的话,那还是用互斥锁吧。

他主要设计到了CAS(compare and swap)操作,如下代码代替了++x;

其中 atomic_compare_exchage_wead 用于比较并交换操作。它用于在原子方式下比较内存中的值&x和给定的期望值&old,如果它们相等,则将新值newval写入内存,并返回 true;否则不写入,并返回 false。

也就是会再去检查内存中x的值,发现是x==old的,证明此时其他线程并没有参与进来,那么你写入新值返回true就完事,如果发现内存中x的值与old不相等,也就不会写入并返回false。

六、shared_ptr的多线程问题

我们知道,多个 shared_ptr 可以共同拥有同一个对象,这里面有一个引用计数。当我们去拷贝shared_ptr的时候,都会对该引用计数进行++操作。

如果是多线程的情况下,去拷贝会不会发生问题呢?

如下代码,本应该对 (*sp)++了20000次,结果却不是20000。

 当我们对资源加锁后,发现*sp的值就是我们预想的了。

由此可得出结论:shared_ptr本身是线程安全的,但是他保护的资源不是线程安全的

七、懒汉模式中的线程安全问题

懒汉模式是在需要时才会创建对象实例,而不是在程序启动时就创建,因此我们之前只有一个执行流执行的时候,只需要判断他的成员变量指针 _instance 是否为空就可以了,为空就创建再返回,不为空就直接返回该指针。

但如果是多线程的情况,可能有很多线程一起起来访问,可能会执行很多new Singleton()。导致数据不一致问题,因此我们得进行加锁。

但如果仅仅是加锁,那么每次线程调用GetInstance()的时候都要去申请锁,效率会很低下。

因此我们可以再在最外层判断一下_instance是否为nullptr,双重保险,让效率提升,如果不为nullptr,那么就直接返回,不用再申请锁了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/563738.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

​「Python大数据」VOC数据清洗

前言 本文主要介绍通过python实现数据清洗、脚本开发、办公自动化。读取voc数据,存储新清洗后的voc数据数据。 一、业务逻辑 读取voc数据采集的数据批处理,使用jieba进行分词,去除停用词,清洗后的评论存储到新的列中保存清洗后的数据到新的Excel文件中二、具体产出 三、执…

three.js(2):渲染第一个three.js三维对象

这一章渲染一个立方体对象到场景中&#xff0c;效果如下&#xff1a; 代码如下&#xff1a; <!DOCTYPE html> <html><head><meta charset"utf-8" /><title></title><script text"module" charset"UTF-8"…

LabVIEW学习记录2 - MySQL数据库连接与操作

LabVIEW学习记录2 - MySQL数据库连接与操作 一、前期准备1.1 windows下安装MySQL的ODBC驱动 二、LabVIEW创建MySQL 的UDL文件三、LabVIEW使用UDL文件进行MySQL数据库操作3.1 建立与数据库的连接&#xff1a;DB Tools Open Connection.vi3.2 断开与数据库的连接&#xff1a;DB T…

功能测试前景揭秘:会被淘汰吗?

在当今快速发展的信息时代&#xff0c;软件已经成为我们工作、学习乃至生活中不可或缺的一部分。随着技术的不断进步和应用的广泛普及&#xff0c;软件测试作为保障软件质量和功能实现的关键步骤&#xff0c;其职业发展路径也受到了广泛的关注。特别是针对功能测试这一细分领域…

The layered MVP architecture in Acise

Acise是一款CAx软件开发平台&#xff0c;本文给出Acise中的MVP架构模式的实现思路。 注1&#xff1a;文章内容会不定期更新。 MVP Data Model View Model 参考文献 Erich Gamma. Design Patterns:elements of reusable object-oriented software. Addison Wesley, 1994.Josep…

高达27K star!基于LLM构建本地智能知识库 太猛了

觉得搞一个AI的智能问答知识库很难吗&#xff1f;那是你没有找对方向和工具&#xff0c; 今天我们分享一个开源项目&#xff0c;帮助你快速构建基于Langchain 和LLM 的本地知识库问答&#xff0c;在GitHub已经获得27K star&#xff0c;它就是&#xff1a;Langchain-Chatchat L…

摄影的技术和艺术,摄影师的日常修养

一、资料描述 本套摄影师资料&#xff0c;大小1.50G&#xff0c;共有67个文件。 二、资料目录 《1900&#xff0c;美国摄影师的中国照片日记》.pdf 《40幅引人入胜的获奖照片》.pdf 《把你的照片换成钱&#xff1a;图片库摄影师的生存之道》(美)陈小波.扫描版.PDF 《半小…

OceanBase数据库日常运维快速上手

这里为大家汇总了从租户创建、连接数据库&#xff0c;到数据库的备份、归档、资源配置调整等&#xff0c;在OceanBase数据库日常运维中的操作指南。 创建租户 方法一&#xff1a;通过OCP 创建 确认可分配资源 想要了解具体可分配的内存量&#xff0c;可以通过【资源管理】功…

百度最新AI旋转验证码

一、简介 先来说说百度旋转验证码的历史。 1、百度旋转验证码 这是百度最早的旋转验证码&#xff0c;只有有限的数量&#xff0c;图片以风景为主&#xff0c;没有随机阴影&#xff0c;没有干扰线条等。所以这种验证码识别比较简单&#xff0c;正确率在99%左右。如下图所示 2…

卸载微软的浏览器: Edge

前言&#xff1a; Edge 崩溃了&#xff0c;无法访问网路&#xff1a; 错误代码: STATUS_STACK_BUFFER_OVERRUN 然后&#xff0c;windows不提供卸载&#xff0c;这下好了&#xff0c;它不能用&#xff0c;你也不能卸载&#xff0c;重新安装也无法解决&#xff0c;咋办&#xff…

hadoop编程之词频统计

数据集实例 java代码&#xff0c;编程 实例 我们要先创建三个类分别为WordCoutMain、WordCoutMapper、WordCoutReducer这三个类 对应的代码如下 WordCoutMain import java.io.IOException; import org.apache.hadoop.conf.Configuration; import org.apache.hadoop.fs.Pat…

亚马逊---设计安全架构

会从以下三个方面展开&#xff1a; 1、AWS资源访问安全 2、应用程序负载的网络安全 3、云中数据的安全 责任共担模式 就像租房子&#xff08;房东和你的责任&#xff09; AWS资源访问安全 需要掌握以下几点&#xff1a; 1、跨多个账户的访问控制和管理 2、AWS联合访问和身份服…

买了个三星i9300(S3)供以后给黑莓Q10开发软件用(安卓4.3)

买了个三星i9300(S3)供以后给黑莓Q10开发软件用(安卓4.3) 前段时间的时候一心想给黑莓Q10开发个软件用用&#xff0c;开发到一半因为过程太过繁琐才叫停了。 一、黑莓Q10安卓应用开发为什么繁琐&#xff1f; Q10的开发过程是这样的&#xff1a; 因为黑莓Q10 里面运行的是Andr…

深度学习之视觉特征提取器——VGG系列

VGG 提出论文&#xff1a;1409.1556.pdf (arxiv.org) 引入 距离VGG网络的提出已经约十年&#xff0c;很难想象在深度学习高速发展的今天&#xff0c;一个模型能够历经十年而不衰。虽然如今已经有VGG的大量替代品&#xff0c;但是笔者研究的一些领域仍然有大量工作选择使用VG…

layabox手游全面屏、ipad屏幕适配方案

1设置 手游平台在项目设置中&#xff0c;场景适配模式选择”固定宽模式 fixedwidth“&#xff0c;设计宽度以全面屏比例为主&#xff0c;我这里设置的设计宽高为640 * 1386 2代码和场景 laya的UI面板有三种类型&#xff0c;分别是Scene、View和Dialog 1&#xff09;Scene和V…

Java web应用性能分析服务端慢之Nginx慢

一般Nginx作为整个应用的入口&#xff0c;即做静态服务器&#xff0c;也做负载均衡、反向代理&#xff1b;同时也因为位置靠前&#xff0c;还可以通过Nginx对于访问的IP、并发数进行相应的限制。在Java web应用性能分析中&#xff0c;Nginx是重要环节&#xff0c;Nginx的性能也…

Flink Job提交分析

1.概述 Flink 应用程序的提交方式为&#xff1a;打成jar包&#xff0c;通过 flink 命令来进行提交。 flink 命令脚本的底层是通过 java 命令启动&#xff1a;CliFrontend 类 来启动 JVM 进程&#xff0c;执行任务的构造和提交。 flink run xxx.jar class arg1 arg2flink.sh 脚…

Springboot+Vue项目-基于Java+MySQL的影城管理系统(附源码+演示视频+LW)

大家好&#xff01;我是程序猿老A&#xff0c;感谢您阅读本文&#xff0c;欢迎一键三连哦。 &#x1f49e;当前专栏&#xff1a;Java毕业设计 精彩专栏推荐&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb; &#x1f380; Python毕业设计 &…

Unity 新版输入系统(Input System)

前言 官方教程 注意 新的输入法系统需要 Unity 2019.4 和 .NET 4 运行时。它不适用于 .NET 3.5 的项目。 教程版本&#xff1a;Unity 2021.3.26 1. 安装 1.1 打开 Package Manager 导航栏 -> Window -> Package Manager 1.2 安装 Input System 选择 Unity Registry 在…

【WEB前端2024】开源元宇宙:乔布斯3D纪念馆-第8课-新增摆件

【WEB前端2024】开源元宇宙&#xff1a;乔布斯3D纪念馆-第8课-新增摆件 使用dtns.network德塔世界&#xff08;开源的智体世界引擎&#xff09;&#xff0c;策划和设计《乔布斯超大型的开源3D纪念馆》的系列教程。dtns.network是一款主要由JavaScript编写的智体世界引擎&#…