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# 比较两个list 之间元素差异的常用方法

《C#比较两个list之间元素差异的常用方法》:本文主要介绍C#比较两个list之间元素差异,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. 使用Except方法2. 使用Except的逆操作3. 使用LINQ的Join,GroupJoin

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

深度解析Java项目中包和包之间的联系

《深度解析Java项目中包和包之间的联系》文章浏览阅读850次,点赞13次,收藏8次。本文详细介绍了Java分层架构中的几个关键包:DTO、Controller、Service和Mapper。_jav... 目录前言一、各大包1.DTO1.1、DTO的核心用途1.2. DTO与实体类(Entity)的区别1

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

Redis出现中文乱码的问题及解决

《Redis出现中文乱码的问题及解决》:本文主要介绍Redis出现中文乱码的问题及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1. 问题的产生2China编程. 问题的解决redihttp://www.chinasem.cns数据进制问题的解决中文乱码问题解决总结

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

c++中的set容器介绍及操作大全

《c++中的set容器介绍及操作大全》:本文主要介绍c++中的set容器介绍及操作大全,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录​​一、核心特性​​️ ​​二、基本操作​​​​1. 初始化与赋值​​​​2. 增删查操作​​​​3. 遍历方

解析C++11 static_assert及与Boost库的关联从入门到精通

《解析C++11static_assert及与Boost库的关联从入门到精通》static_assert是C++中强大的编译时验证工具,它能够在编译阶段拦截不符合预期的类型或值,增强代码的健壮性,通... 目录一、背景知识:传统断言方法的局限性1.1 assert宏1.2 #error指令1.3 第三方解决