49 C++ 多个线程之间共享资源问题。lock , unlock

2024-01-15 09:12

本文主要是介绍49 C++ 多个线程之间共享资源问题。lock , unlock,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前提,我们要补充一个知识点。再使用类成员函数做为 线程启动的入口,第二个参数可以传递对象 和 对象地址,如下:

类似这样:thread readthread(&Teacher164::readfunc,tea);thread readthread(&Teacher164::readfunc,&tea);

那么 这两种 有啥区别?

//当我们在构造一个 thread的时候,如下代码对应 readthread,writethread,
//如果第二个参数传递的是 Teacher164对象 tea,那么会有copy 构造函数的调用。


//如果 我们在传递第二个参数的时候, 传递的是Teacher164对象的tea的地址,相当于多个线程中都会共享这个tea,这才会让在 Teacher164类中的list是同一个,才有共享数据的可能性。

class Teacher164 {public://共享数据 存在list中list<int> msgListQueue;
public://读取 共享数据的线程方法void readfunc() {int readcount = 0;while (true && readcount!=100000) {//只要不为空,就可以读取数据if (!msgListQueue.empty()) {int readvalue = msgListQueue.front();//每次都读取第一个cout << "读取到的值为" << readvalue << endl;msgListQueue.pop_front();//删除第一个元素readcount++;}else {cout << "没有读取到值" << endl;}}}//写入 共享数据的线程方法void writefunc() {for (size_t i = 0; i < 100000; i++){msgListQueue.push_back(i);//每次都写到末尾cout << "写入元素的值为" << i << endl;}}public:Teacher164() {cout << "Teacher164 构造方法 this = " << this << endl;}Teacher164(const Teacher164 & obj) {cout << "Teacher164 copy 构造方法 this = " << this << "  obj = " << &obj<< endl;}~Teacher164() {cout << "Teacher164 析构函数 this = " << this << endl;}
};//当我们在构造一个 thread的时候,如下代码对应 readthread,writethread,
//如果第二个参数传递的是 Teacher164对象 tea,那么会有copy 构造函数的调用。
//如果 我们在传递第二个参数的时候, 传递的是Teacher164对象的tea的地址,相当于多个线程中都会共享这个tea,这才会让在 Teacher164类中的list是同一个,才有共享数据的可能性。void main() {Teacher164 tea;thread readthread(&Teacher164::readfunc,tea);thread writethread(&Teacher164::writefunc, tea);readthread.join();writethread.join();//如上,调用的时候,tea是值传递的时候,会调用 copy 构造函数//Teacher164 构造方法 this = 00000092A277F508//	Teacher164 copy 构造方法 this = 000002713BA6C420  obj = 00000092A277F508//	Teacher164 copy 构造方法 this = 000002713BA75310  obj = 00000092A277F508//	Teacher164 析构函数 this = 000002713BA6C420//	Teacher164 析构函数 this = 000002713BA75310//	Teacher164 析构函数 this = 00000092A277F508cout << "=========================" << endl;
}

//传递 是地址:  thread readthread1(&Teacher164::readfunc, &tea1); 

结果是:只要使用的 &tea1 的线程,都使用的同一个tea(地址是一样的,没有copy 构造函数调用),这就有可能让其共享数据

void main(){Teacher164 tea1;thread readthread1(&Teacher164::readfunc, &tea1);thread writethread1(&Teacher164::writefunc, &tea1);//thread 的构造方法第二个参数 可以是地址,如果是地址,那么 readthread1 和 writethread1的传递就是同一个 teacher//  Teacher164 构造方法 this = 000000265F4FFB68//	Teacher164 析构函数 this = 000000265F4FFB68}

对于只读的资源,所有线程都可以一起去读,这是线程安全的。

但是如果有些线程是读资源,有些线程是写资源,那么就要注意线程之间的资源共享问题了。

//我们先把问题简化,假设有2个线程,一个线程 读数据,一个线程写数据。但是运行确一直没有问题。
//于是搞4个线程,2个读数据,2个写数据

有问题的代码

class Teacher164 {public://共享数据 存在list中list<int> msgListQueue;
public://读取 共享数据的线程方法void readfunc() {int readcount = 0;while (true && readcount!=100000) {//只要不为空,就可以读取数据if (!msgListQueue.empty()) {int readvalue = msgListQueue.front();//每次都读取第一个cout << "读取到的值为" << readvalue << endl;msgListQueue.pop_front();//删除第一个元素readcount++;}else {cout << "没有读取到值" << endl;}}}//写入 共享数据的线程方法void writefunc() {for (size_t i = 0; i < 100000; i++){msgListQueue.push_back(i);//每次都写到末尾cout << "写入元素的值为" << i << endl;}}public:Teacher164() {cout << "Teacher164 构造方法 this = " << this << endl;}Teacher164(const Teacher164 & obj) {cout << "Teacher164 copy 构造方法 this = " << this << "  obj = " << &obj<< endl;}~Teacher164() {cout << "Teacher164 析构函数 this = " << this << endl;}
};void main() {cout << "=========================" << endl;Teacher164 tea1;thread readthread1(&Teacher164::readfunc, &tea1);thread writethread1(&Teacher164::writefunc, &tea1);//thread 的构造方法第二个参数 可以是地址,如果是地址,那么 readthread1 和 writethread1的传递就是同一个 teacherthread readthread2(&Teacher164::readfunc, &tea1);thread writethread2(&Teacher164::writefunc, &tea1);readthread1.join();readthread2.join();writethread1.join();writethread2.join();}

fix方案,使用mutex(锁子),结合lock() 和 unlock()方法

这篇关于49 C++ 多个线程之间共享资源问题。lock , unlock的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/608383

相关文章

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

JDK21对虚拟线程的几种用法实践指南

《JDK21对虚拟线程的几种用法实践指南》虚拟线程是Java中的一种轻量级线程,由JVM管理,特别适合于I/O密集型任务,:本文主要介绍JDK21对虚拟线程的几种用法,文中通过代码介绍的非常详细,... 目录一、参考官方文档二、什么是虚拟线程三、几种用法1、Thread.ofVirtual().start(

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

IDEA和GIT关于文件中LF和CRLF问题及解决

《IDEA和GIT关于文件中LF和CRLF问题及解决》文章总结:因IDEA默认使用CRLF换行符导致Shell脚本在Linux运行报错,需在编辑器和Git中统一为LF,通过调整Git的core.aut... 目录问题描述问题思考解决过程总结问题描述项目软件安装shell脚本上git仓库管理,但拉取后,上l

idea npm install很慢问题及解决(nodejs)

《ideanpminstall很慢问题及解决(nodejs)》npm安装速度慢可通过配置国内镜像源(如淘宝)、清理缓存及切换工具解决,建议设置全局镜像(npmconfigsetregistryht... 目录idea npm install很慢(nodejs)配置国内镜像源清理缓存总结idea npm in

pycharm跑python项目易出错的问题总结

《pycharm跑python项目易出错的问题总结》:本文主要介绍pycharm跑python项目易出错问题的相关资料,当你在PyCharm中运行Python程序时遇到报错,可以按照以下步骤进行排... 1. 一定不要在pycharm终端里面创建环境安装别人的项目子模块等,有可能出现的问题就是你不报错都安装

Java 线程池+分布式实现代码

《Java线程池+分布式实现代码》在Java开发中,池通过预先创建并管理一定数量的资源,避免频繁创建和销毁资源带来的性能开销,从而提高系统效率,:本文主要介绍Java线程池+分布式实现代码,需要... 目录1. 线程池1.1 自定义线程池实现1.1.1 线程池核心1.1.2 代码示例1.2 总结流程2. J

idea突然报错Malformed \uxxxx encoding问题及解决

《idea突然报错Malformeduxxxxencoding问题及解决》Maven项目在切换Git分支时报错,提示project元素为描述符根元素,解决方法:删除Maven仓库中的resolv... 目www.chinasem.cn录问题解决方式总结问题idea 上的 maven China编程项目突然报错,是

Python爬虫HTTPS使用requests,httpx,aiohttp实战中的证书异步等问题

《Python爬虫HTTPS使用requests,httpx,aiohttp实战中的证书异步等问题》在爬虫工程里,“HTTPS”是绕不开的话题,HTTPS为传输加密提供保护,同时也给爬虫带来证书校验、... 目录一、核心问题与优先级检查(先问三件事)二、基础示例:requests 与证书处理三、高并发选型: