线程安全问题(3)

news2025/1/9 16:58:39

线程不安全:在多线程的调度情况下,导致出现了一些随机性,随机性是代码中出现了一些BUG,导致我们的线程是不安全的

造成线程不安全的原因:

1)操作系统抢占式执行,线程调度随机,这是万恶之源,我们无能为力

2)多个线程同时修改同一个变量,适当调整代码结构,避免这种情况

3)针对的变量操作不是原子的,count++本质上是三个指令:load,add,save

4)内存可见性:一个线程读,一个线程写,直接读寄存器,不读内存,这也是一种优化

5)指令重排序:编译器优化,CPU自身的执行

在这里面,涉及到一个重要的知识点,JMM(java memory model内存模型)

1)我们正常情况下,是想要把内存中的数据读到CPU里面进行操作,但是实际上在CPU和内存中还会存在这一些缓存,为了提高CPU和内存之间的读写效率

2)当我们在代码中读取变量的时候,不一定是在真的读内存,可能这个数据已经在CPU或者cathe中缓存着了
3)这个时候就可能会绕过内存,直接从CPU寄存器里面或者cathe中取数据;咱们的JMM针对计算机的硬件结构,又进行了一层抽象,主要是因为Java是要跨平台,要能够支持不同的计算,有的计算机可能没有catche2内存,可能也会有catche3内存,
当这个变量进行修改的时候,此时读的这个线程没有从内存里面读,而是从catche里面读,或者CPU中的寄存器里面读,就会出现内存可见性的问题;

4)KMM就会把CPU中的寄存器,L1,L2,L3catche,统称为工作内存;把真正的内存称为主内存,工作内存一般不是真的内存,每一个线程都会有自己的工作内存,每个线程都有独立的上下文,独立的上下文就是各自的一组寄存器/catche中的内容
5)CPU和内存进行交互时,经常会把主内存中的内容,拷贝到工作内存,然后再进行工作,写会到主内存中,这就可能会出现数据不一致的情况,这种情况是在编译器进行优化的时候特别严重,关键字volitaile和synchronized就可以强制保证接下来的操作是在操作内存,在生成的java字节码中强制插入一些内存屏障的指令,这些指令的效果,就是强制刷新内存,同步更新主内存和工作内存中的内容,在牺牲效率的时候,保证了准确性
注意:synchronized与voilatile的区别
他们都可以保证内存可见性,但是synchronized可以保证原子性,volitaile不可以保证原子性

1)计算机想要执行一些计算,就需要把内存中的数据写入到CPU的寄存器里面,然后再在寄存器中进行计算,再写回到内存里面,直接读寄存器,CPU访问寄存器的速度是要比访问内存的速度要快的多,当我们的CPU连续多次访问内存,发现结果都一样,就不会从内存里面读了,就直接会从CPU的寄存器里面读就可以了;

2)在Java中,我们要先把数据从主内存加载到工作内存里面,工作内存计算完毕之后在写回到主内存里面

3)咱们的类加载双亲委派模型就是描述了先去哪一个目录找.class,后去哪一个目录中找

4)CPU从内存中取数据,取得太慢了,尤其是频繁进行取的时候,我们就可以把这样的数据放到寄存器里面,然后直接从寄存器里面读,但是我们的寄存器空间是很紧张的

5)于是我们的CPU又搞了一个内存空间,这个空间比寄存器大,比内存小,速度比寄存器慢,比内存快,我们就称之为缓存

1.存储空间:CPU<L1<L2<L3<内存

2.速度:CPU>L1>L2>L3>内存

3.成本:CPU>L1>L2>L3>内存

最最常用的数据放到CPU寄存器里面,其次常用的放到L1里面,再次常用的放到L2里面

我们的可重入锁的意义就是降低了程序员的负担(提高了开发效率)

缓存+CPU寄存器=CPU

6)咱们的synchronized的使用是要付出代价的,代价就是一旦使用synchronized很容易会导致线程阻塞,一旦线程阻塞(放弃CPU),下一次我们再次回到CPU就会变得很困难,这个时间是不可控的,如果调度不回来,自然对应的任务执行时间也就被拖慢了,使用synchronized就会和高性能无缘,volatile不会使线程阻塞

 但是我们的程序也会有更高的开销(维护锁属于哪一个线程,并且进行引用计数,降低了运行效率)

如果我们的使用的是不可重入锁,此时我们的开发效率就低了,一不小心,咱们的代码就很容易写出BUG,如果BUG严重,就会造成严重后果

3.标准库中的集合类大部分是线程不安全的

ArrayList/LinkedList/HashMap/HashSet/Treeset/StringBuilder都是线程不安全的;

线程安全:Vector(JDK早期内置的一个集合类,这里面的设计并不是特别合理,他也是一个顺序表,是一个动态数组,他这里面使用了synchronized来修饰了很多方法,大多数情况下加上synchronized就会使单线程环境下的操作的效率造成负面影响;会禁止编译器的优化,对标ArrayList);

HashTable线程安全,不建议使用

Stack线程安全,ConcurrentHashmap,StringBuffer

String是线程安全的,没有进行加锁,String是不可变对象,不可能存在两个线程同时修改一个相同的String对象,没有提供对public 的修改属性的操作(修改char[]数组)

而final表示String不能被继承

我们使用线程安全的集合类,那么有了这个操作,我们就可以保证在多线程环境下,我们修改同一个对象,就不会有太大的问题

 

Synchronized(也叫做监视器锁)有什么用处呢?

1)保证操作是原子的,互斥
2)synchronized不光可以起到互斥的效果,还能够刷新内存,解决内存可见性的问题
例如在一个代码中循环的进行++操作,每次自增,都不是原子的,编译器会优化这里面的效率,把从内存中读取数据到CPU中,和把CPU中的数据放回到内存中这些过程会省略;
2.1)加上synchronized之后,就会优化上面的操作,保证把数据从内存里面读,也会真正的把数据写回到内存
2.2)也是让程序跑的慢一点,但是能够算得准,一旦代码中使用了synchronized,此时咱们的程序就可能与高性能无缘了,在想跑快就很困难了;
2.3)本来在单线程中,我们是可以进行优化的,减少访问内存的操作,但是加上synchronized之后,就必须强制访问内存的数据到寄存器里面;
3)避免可重入:synchronized允许一个线程针对同一把锁,咔咔加锁两次,如果出现了死锁,那么就是不可重入的,如果是不会发生死锁,那么就是可重入的锁

synchronized public void increase(){
     synchronized(this){
              count++;
   }
}

上面这种情况就是说外层先针对当前对象加了一次锁,在里层又对这个对象再加了一次锁,这就是一次锁两次

1)外层锁:进入方法之后,就开始进行加锁,这次我们能够加锁成功,因为当前锁是没有其它线程进行占用的

2)里层锁:当我们对外层方法进行加锁之后,进入代码块之后,开始尝试进行加锁,这一次加锁是不能够加锁成功的,因为按照之前的观点进行分析,锁在外层是被占用这呢,只有当前持有锁的方法执行完之后,我们才可以获取到锁;外层锁只有执行完整个方法,才可以释放锁,但是想要执行完整个方法,我们就需要让里层锁加锁成功之后继续走下去

3)所以我们的锁就永远无法释放,不会有任何的线程可以获取到锁了,就发生了死锁

死锁:是指多个进程在运行过程中因争夺资源而造成的一种僵局,当进程处于这种僵持状态时,若无外力作用,它们都将无法再向前推进
1)进入increase之后加了一次锁,进入代码块之后又加了一锁,在这里面synchronized在这里进行了特殊处理;
2)如果是其他语言的锁操作,就有可能造成死锁;
第一次加锁,加锁成功
第二次在尝试对这个对象头加锁的时候,此时对象头的锁标记已经是true,按照之前的理解,线程就要进行阻塞等待,等待这个所标记被改成false,才重新竞争这个锁,但是此时是在方法内部,肯定是无法释放第一次所加的锁的,就出现死锁了;

synchronized public void increase()
{  
     count++;
}

synchronized public void increase2()
{
   increase();
}
  synchronized public static  void run()
    {
        synchronized (Student.class) {
            System.out.println("我叫做run方法");
        }
    }

3)在可重入锁内部,会记录当前的锁是被哪一个线程占用的,同时也会记录一个加锁次数3.1)当我们的线程A对这个锁进行第一次加锁的时候,显然是能加锁成功的,锁内部就会进行记录,当前占用着的线程是A,同时加锁次数是1,后续我们再次针对这个线程A进行加锁的时候,此时就不会真的加锁,而只是单纯的把引用计数给自增,加锁次数是2,后续我们进行解锁的时候,再把引用计数减1,当我们进行把引用计数减到0的时候,就真的进行解锁所以说后续的加锁操作对这个线程持有这把锁是没有本质影响的;

3.2)咱们的可重入锁的意义就是为了降低程序员的负担,降低使用成本,提高了开发效率,但是也带来了代价,程序中需要有额外的开销,因为我们要维护锁属于哪一个线程,并且进行加减计数,这个变量还占用空间,降低了运行效率,开发效率最重要

3.3)如果说我们使用的是不可重入锁,此时我们的开发效率就降低了,如果我们一不小心,代码中就出现死锁了,线上程序出BUG了,就需要修BUG了,如果BUG严重了,年终奖可能会泡汤

4)解决指令重排序,避免乱序执行

出现死锁的其他情况:

1)一个线程针对一把锁咔咔枷锁两次

2)两个线程,两把锁

比如说我和我朋友去吃饺子

我:吃饺子习惯蘸酱油

朋友:吃饺子习惯蘸醋

2.1)当我拿起了酱油之后,我的朋友拿起了醋,我说:请你把醋给我,朋友说:请你把酱油给我

2.2)我说:你先把醋给我,我用完了,我就给酱油,朋友说:你先把酱油给我,我用完了就给你醋,我们在这里面可知,醋和酱油就是两把锁

3)N个线程,M把锁

synchronized(A){
  synchronized(B){
    synchronized(c){
}
 }
  }

实际上咱们如果说不使用嵌套锁,那么也不会容易出现死锁,如果咱们的适应场景,不得不进行嵌套,那么我们一定要注意约定好加锁的顺序,所有的线程都用a-b-c这样的顺序进行加锁,千万别的线程用c-a-b的方式进行加锁,或者是a-c-b,否则就会很容易发生环路等待

哲学家共餐问题:

 

1)我们的每一个哲学家什么时候思考人生,什么时候吃面条,是不确定的,每一个哲学家就对应着我们的线程,这样的思考人生和吃鸡的随机性也就体现着我们线程调度的随机性;

2)我们的哲学家进行吃面条的时候,都需要拿起来他身边的两根筷子,假设先拿起左手的筷子,再拿起来右手边的筷子

3)咱们的哲学家是非常固执的,如果想吃面条的时候,尝试拿筷子的时候,发现筷子一直被别人占用着,就会一直等,如果在这个模型中如果五个哲学家,同时伸出左手,拿起来左手边的筷子,就会发生死锁问题

4)每一个人都成功的拿起来了左手边的筷子,但是永远无法成功拿起来右手边的筷子,而我们的哲学家又是非常固执地,不肯放下自己手中的筷子,那么所以说最后谁也不会成功吃到面条

咱们只需要约定好,针对多把锁进行加锁的时候,我们有固定的顺序就好了,当我们所有的线程都遵守同样的规则顺序,就不会出现环路等待;

解决哲学家共餐问题:

1)我们的解决方法就是给当前所有的筷子都编上号,我们约定让哲学家拿筷子(同时拿筷子),不是先拿左手,再拿右手

2)而是哲学家先拿编号小的,再拿编号大的(咱们的哲学家是非常固执地,只会拿他身旁的两个筷子中的较小的那个,即使较小的那个筷子被别人拿了,也会等待,虽然旁边有编号较大的筷子,但是还是优先拿起编号小的筷子)

总结:死锁的四个必要条件 

1)互斥使用:一个线程被另一个线程占用了之后,其他线程无法进行占用,锁的本质,保证原子性

2)不可抢占,不可剥夺性:一把锁被另一个线程占用了之后,其他的线程是无法把这把锁给抢走的

3)请求和保持:当一个线程占据了多把锁之后,除非显示的进行释放锁,否则这些锁都是该线程所持有的

4)环路等待,在我们的实际开发中,如果说想要避免死锁,那么我们关键要点还是从第四个条件来进行切入,

当synchronized修饰方法的时候有以下需要进行注意:

1)synchronized关键字不可以被继承:虽然我们可以使用synchronized来进行修饰方法,但是synchronized并不属于方法中的一部分,因此synchronized关键字不可以被继承,如果说你在父类中的某一个方法中使用了synchronized关键字,在子类中重写了这个方法,在子类中的某一个方法并不是同步的,必须显示的在子类中加上synchronized关键字才可以

2)定义接口方法中不能使用synchronized关键字

3)在构造方法中不能使用synchronized关键字,但是可以使用synchronized同步代码快来进行同步

总结:synchronized修饰普通方法和同步代码快指定this表示给当前对象进行加锁,就是当不同线程尝试访问一个对象中的synchronized(this)同步代码快的时候,其他访问该对象的线程将会被阻塞

class RunnableTask implements Runnable{
    public void GetCount(){
System.out.println("生命在于运动");
  }
    public void run() {
        synchronized (this){
            try {
                TimeUnit.SECONDS.sleep(10);
                System.out.println("我是中国人");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}
class Solution {
    public static void main(String[] args) {
        RunnableTask task=new RunnableTask();
        Thread t1=new Thread(task);
        Thread t2=new Thread(task);
        t1.start();
        t2.start();
        //上面这种情况就会发生阻塞
        //但是下面这种情况就不会发生阻塞,不会产生锁的竞争
        RunnableTask task1=new RunnableTask();
        RunnableTask task2=new RunnableTask();
        Thread t3=new Thread(task1);
        Thread t4=new Thread(task2);
        t3.start();
        t4.start();

    }
}

解释:

1)当我们的两个并发线程t1和t2在进行访问同一个对象的synchronized(this)修饰的同步代码快的时候,同一时刻只能有一个线程被执行,一个线程被阻塞,必须等待一个线程执行玩这个同步代码快之后另一个线程才可以执行这个代码块

2)t1和t2是互斥的,因为在我们执行synchronized代码块的时候会进行锁定当前的对象,只有执行完该同步代码快的才能释放该对象的锁,下一个线程才能执行并且锁定该对象

3)但是被注释掉的那一片代码,t3和t4在同时进行执行,因为他们是访问两个对象中的被synchronized修饰的方法或者是同步代码快,这是因为synchronized只锁定对象,每一个对象只有一把锁与之相关联

4)当一个线程访问一个对象的synchronized的同步代码快的时候,另一个线程仍然可以访问该对象的非同步代码快

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Web进阶:Day7 响应式、BootStrap、实战演练

Web进阶&#xff1a;Day7 Date: January 10, 2023 Summary: 响应式、BootStrap、实战演练 响应式 媒体查询 目标&#xff1a;能够根据设备宽度的变化&#xff0c;设置差异化样式 媒体特性常用写法 媒体特性常用写法&#xff1a; max-width&#xff08;从小到大&#xff0…

transformers包介绍——nlp界最顶级的包——可以不用 但不能不知道——python包推荐系列

背景1 现在在AI行业&#xff0c;什么最火&#xff1f;计算机视觉还是自然语言处理&#xff1f;其实不得不说&#xff0c;现在nlp很火。还有人记得上个月很多科技爱好者都在玩的chatgpt么&#xff1f;那个就是nlp技术的一大应用。现在都在觉得AI赚钱&#xff0c;工资高&#xf…

深度学习 12 正则化

1. 对于高方差(过拟合)&#xff0c;有以下几种方式&#xff1a; 获取更多的数据&#xff0c;使得训练能够包含所有可能出现的情况 正则化&#xff08;Regularization&#xff09; 寻找更合适的网络结构 2. 对于高偏差(欠拟合)&#xff0c;有以下几种方式&#xff1a; 扩大网…

【C语言进阶】只看此篇,让你学会动态内存管理

目录 前言 一、为什么存在动态内存分配 二、动态内存函数的介绍 1 、malloc和free 2、 calloc 3 、realloc 三、常见的动态内存错误 四、动态内存管理笔试题 1 题目1&#xff1a; 2 题目2&#xff1a; 3 题目3&#xff1a; 4 题目4&#xff1a; 五、C/C程序的…

5.10回溯法--圆排列问题--排列树

圆排列问题描述 给定n个大小不相等的圆&#xff0c;要将这n个大小不相等的圆排进一个矩形框中&#xff0c;且要求个个圆都与矩形框的最底边相切。要找出最小长度的圆排列。 问题分析 排列排列&#xff0c;解空间是一个排列树。 设开始时&#xff0c;a[n]储存n个圆的半径&…

笔试强训48天——day26

文章目录一. 单选1.在单处理器系统中&#xff0c;如果同时存在有12个进程&#xff0c;则处于就绪队列中的进程数量最多为&#xff08;&#xff09;2.在系统内存中设置磁盘缓冲区的主要目的是&#xff08;&#xff09;3.下列选项中&#xff0c;会导致进程从执行态变为就绪态的事…

时间复杂度空间复杂度

算法效率数据结构算法时间复杂度大O的渐进表示法三种时间复杂度几道例题一.简单递归二结合代码来判断时间复杂度空间复杂度我们程序猿开始的时候肯定听了不少的:数据结构和算法,那么阿涛就给大家说说自己的拙见吧.数据结构 数据结构就是我们用来组织数据的方式,比如我们可以把…

JVM-【面试题】-垃圾收集算法+垃圾收集器,以后就不用担心对象那些事了

一、垃圾收集算法在jvm里对可回收的对象在不同的垃圾收集器里&#xff0c;有不同的回收算法&#xff0c;具体的可以分为这四种&#xff1a;分代收集算法、复制算法、标记清除算法、标记整理算法1.1 分代收集算法当前虚拟机的垃圾收集都采用分代收集算法&#xff0c;这种算法没有…

【NI Multisim 14.0操作实例——最小锁存器电路】

目录 序言 &#x1f525;1.设置工作环境 &#x1f525;2.设置原理图图纸 &#x1f525; 3.设置图纸的标题栏 &#x1f525; 4. 增加元器件 &#x1f525; 5. 放置总线 &#x1f525; 6. 添加总线分支 序言 NI Multisim最突出的特点之一就是用户界面友好。它可以使电路…

ELasticSearch监控之Cerebro安装

文章目录安装包下载安装使用点击nodes 查看各节点状态点击rest发送可以发送请求查询数据安装包下载 https://github.com/lmenezes/cerebro/releases/download/v0.9.4/cerebro-0.9.4.tgzgit地址&#xff1a;https://github.com/lmenezes/cerebro 安装 将要包移动到/opt目录解…

Unity 3D 导入三维模型||Unity 3D 动画系统简介(Mecanim)

将三维模型导入 Unity 3D 是游戏开发的第一步。 下面以 3ds Max 为例&#xff0c;演示从三维建模软件中将模型导入 Unity 3D 的过程&#xff0c;具体步骤如下。 在 3ds Max 中创建房子模型。执行 Export → Export 命令导出 fbx 模型。设置保存路径以及文件名。选择默认设置选…

【transformers】tokenizer用法(encode、encode_plus、batch_encode_plus等等)

tranformers中的模型在使用之前需要进行分词和编码&#xff0c;每个模型都会自带分词器&#xff08;tokenizer&#xff09;&#xff0c;熟悉分词器的使用将会提高模型构建的效率。 string tokens ids 三者转换 string → tokens tokenize(text: str, **kwargs)tokens → strin…

安卓车机系统adb shell cmd 源码原理分析

hi&#xff0c;粉丝朋友们大家好&#xff01; 上一次视频分享了input专题课中input命令在android 12的更新&#xff0c;因为原来课程是基于android 10 &#xff08;可以加我扣&#xff1a;2102309716 优惠购买&#xff09; https://ke.qq.com/course/package/77595?tuin7d4eb3…

联合证券|金融部门开年推出新方案 改善优质房企资产负债状况

当时&#xff0c;努力促进房地产与金融正常循环是金融部门的工作要点之一。记者日前了解到&#xff0c;为遵循落实中心经济工作会议布置&#xff0c;有用防范化解优质头部房企危险&#xff0c;改进财物负债情况&#xff0c;有关部门起草了《改进优质房企财物负债表计划举动计划…

【计算机体系结构基础】流水线异常处理和提高流水线效率的技术(一)

流水线中的异常 异常的来源 外部事件指令执行中的错误数据完整性的问题地址转换异常系统调用陷入需要软件修正的运算 异常可以分为&#xff1a;可恢复异常和不可恢复异常 不可恢复异常&#xff1a;系统硬件出现严重故障&#xff0c;异常处理后系统面临重启。 解决办法&#x…

Spring Boot Apollo监听namespace并更新配置Bean(附源码)

这里是weihubeats,觉得文章不错可以关注公众号小奏技术&#xff0c;文章首发。拒绝营销号&#xff0c;拒绝标题党 背景 如果我们使用的配置中心是apollo的话我们经常会遇到这样的问题&#xff0c;就是动态更新配置Bean 动态更新配置bean 动态更新配置bean其实是很简单的&…

Python学习笔记-PyQt6消息窗

对话框是界面编程中重要的窗体&#xff0c;一般用于提示或者一些其他特定操作。一、使用QDialog显示通用消息框直接使用QDialog类&#xff0c;可以及通过对话框进行通用对话框显示&#xff0c;亦可以通过自定义设置自己需要的对话框。# _*_ coding:utf-8 _*_import sysfrom PyQ…

MySQL主从复制的原理是什么?

主从复制是指将主数据库的 DDL 和 DML 操作通过二进制日志传到从库服务器中&#xff0c;然后在从库上对这些日志重新执行(也叫重做)&#xff0c;从而使得从库和主库的数据保持同步。 MySQL支持一台主库同时向多台从库进行复制&#xff0c; 从库同时也可以作为其他从服务器的主…

Python批量下载某网站贵得要shi文档 并保存为PDF

人生苦短&#xff0c;我用Python 基本开发环境&#x1f4a8; Python 3.6Pycharm 相关模块的使用&#x1f4a8; import requests import parsel import re import os import pdfkit需要使用到一个软件 wkhtmltopdf 这个软件的作用就是把html文件转成PDF 想要把文档内容保存…

人工智能轨道交通行业周刊-第30期(2023.1.9-1.15)

本期关键词&#xff1a;贵阳智慧车站、城轨智能化汇总、隧道巡检、信创厂商、手语数字人 1 整理涉及公众号名单 1.1 行业类 RT轨道交通中关村轨道交通产业服务平台人民铁道世界轨道交通资讯网铁路信号技术交流北京铁路轨道交通网上榜铁路视点ITS World轨道交通联盟VSTR铁路与…