【JavaEE初阶】第六节.多线程 (基础篇 )线程安全问题(下篇)

news2025/1/12 12:08:08

前言

一、内存可见性

 二、内存可见性的解决办法 —— volatile关键字

 三、wait 和notify 关键字

3.1 wait() 方法

3.2 notify() 方法 

3.3 notify All() 方法

3.4 wait 和 sleep 的对比 

总结


前言

本节内容接上小节有关线程安全问题;本节内容我们将介绍有关内存可见性和wait,notify关键字的有关知识;让我们一起进入到今天的线程安全的学习当中;


提示:以下是本篇文章正文内容,下面案例可供参考

一、内存可见性

就来介绍一下 怎样解决 由于内存可见性而引发的 线程安全问题;

内存可见性 所存在的场景 是:一个线程读、一个线程写的场景;

package thread;
 
import java.util.Scanner;
 
public class Demo16 {
    //写一个 内部类,此时这个内部类 就处在 Demo16 的内部,就可以解决 前面已经写过 Counter 的问题
    static class Counter {
        public int flg = 0;
    }
 
    public static void main(String[] args) {
        Counter counter = new Counter();
        
        Thread t1 = new Thread(() -> {
            while (counter.flg == 0) {
                //执行循环,但是此处循环 啥都不做
            }
            System.out.println("t1循环结束");
        });
        t1.start();
        
        Thread t2 = new Thread(() -> {
            //让用户输入一个数字,赋值给 flg
            Scanner scanner = new Scanner(System.in);
            System.out.println("请输入一个整数:");
            counter.flg = scanner.nextInt();
        });
        t2.start();
    }
}

预期效果:

t2线程 输入一个非零的整数后,此时 t1线程 循环结束,随之进程结束;

运行结果: 

这,就是内存可见性的问题

分析:

t1线程 的工作:

load 读取内存的数据到 CPU 的寄存器;
test 检测 CPU寄存器的值是否和预期的一样;
同时,反复进行,频繁进行;

由于 读内存比读 CPU寄存器 慢上几千倍、上万倍,意味着 t1线程 的主要操作就在 load上,但是 每一次读取到的值又没有啥变化,于是 直接进行了优化,就相当于 只从内存中只读取一次数据,后续就直接从寄存器里面 进行反复 test 就好了;

编译器看到这个线程(t1线程)对变量 flg 也没有做修改,于是就进行了优化操作;

但是,这里出现了一个特殊情况,有其他的线程(t2线程)对这个变量做出了修改;

但是,t1线程 仍然是 采用之前的数据来读寄存器,此时 读到的数据和内存的数据是不一致的,这种情况就叫做 内存可见性问题(即 内存改了,但是线程没有看见;或者说,没有及时读取到内存中的最新数据)??????????????????????????????

 二、内存可见性的解决办法 —— volatile关键字

由于 编译器优化,是属于编译器自带的功能,正常来说,程序员并不好干预;

但是 因为上述的场景,编译器知道自己可能会出现误判,因此就给程序猿提供了一个 干预优化的途径 —— volatile关键字;

这个关键字是写到要修改的变量上,要保证哪个变量的内存可见性 就往哪个变量里面加;

 或者:

 注意: volatile 可以修饰变量的位置,也是在 public 左右;

 此时,运行结果:

 volatile 操作 相当于是 显示得禁止了编译器进行上述优化,相当于是给这个对应的变量加上了 "内存屏障"(特殊的二进制指令),JVM 再读取这个变量的时候,因为内存屏障的存在,就知道每次都要重新读取这个变量的内容,而不是草率的进行优化了;

虽然频繁的读取内存,使得速度变慢了,但是数据却是算的对;


当然,编译器的优化,是根据代码的实际情况来运行的,在一开始的代码中,由于 循环体是空,所以循环的转速极快,导致了 读内存的操作非常频繁,所以就出发了优化;

但是,如果在循环体中加上 sleep,让循环转速一下子就慢了读取内存的操作 就不是特别频繁了,就不会被触发优化了;

package thread;
 
import java.util.Scanner;
 
public class Demo16 {
    //写一个 内部类,此时这个内部类 就处在 Demo16 的内部,就可以解决 前面已经写过 Counter 的问题
    static class Counter {
        public int flg = 0;
    }
 
    public static void main(String[] args) {
        Counter counter = new Counter();
 
        Thread t1 = new Thread(() -> {
            while (counter.flg == 0) {
                //执行循环,此处加上 sleep 操作
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            System.out.println("t1循环结束");
        });
        t1.start();
 
        Thread t2 = new Thread(() -> {
            //让用户输入一个数字,赋值给 flg
            Scanner scanner = new Scanner(System.in);
            System.out.println("请输入一个整数:");
            counter.flg = scanner.nextInt();
        });
        t2.start();
    }
}

运行结果:

所以说,编译器到底什么时候会优化,仍然是一个 "玄学"问题,它内部有一个完整的优化体系,但是也不关咱们啥事;

由于咱们也不好确定 什么时候优化,什么时候不优化,所以还得要在必要的时候加上 volatile;


注意:

  1. volatile关键字 保证的是 内存可见性 的问题,它不保证原子性问题;
  2. volatile 解决的是 一个线程读、一个线程写 的问题;
  3. 当然,volatile 也可以解决指令重排序的问题;
  4. synchronized 保证的是 原子性的问题,解决的是 两个线程写 的问题;

 三、wait 和notify 关键字

前面已经介绍到,线程它是随机调度的,这个随机性很讨厌,我们希望可以控制线程的执行顺序;

我们可以用 join关键字 来控制 线程结束 的顺序了;

但是,我们仍希望 让两个线程按照既定的顺序配合执行;

wait 和 notify 关键字就可以做到这个效果,相比于 jion,它们可以更好的控制线程之间的执行顺序;


wait 叫做 "等待",调用 wait 的线程,就会进入线程阻塞等待的状态(即 Waiting状态);

notify 叫做 "通知 / 唤醒",调用 notify 的线程,就可以把对应的 wait 线程给唤醒(即 从阻塞状态恢复回就绪状态);

说明:

wait 和 notify 都是 Object 的成员方法(随便哪个对象都可以调用);

比如说:

如果有 o1.wait();

那么 o1.notify()就可以唤醒调用 o1.wait() 的线程,而 o2.notify() 是不能够唤醒调用 o1.wait() 的线程的;


3.1 wait() 方法

wait() 内部的执行过程:

  1. 释放锁
  2. 等待通知
  3. 当通知到达后,就会被唤醒,并且尝试重新获取锁

注意:

wait() 一上来就要释放锁,这就说明 在调用 wait 之前,就需要先拿到锁;

换句话说,wait 必须要放到 synchronized 中使用,并且 synchronized 加锁的对象 和 调用 wait 方法的对象 是同一个对象;

此时,使用 object.wait() 之后就会一直等待下去,但是程序肯定不会这么一直等待下去了,所以这个时候就需要一个唤醒的方法 —— notify()  ;

3.2 notify() 方法 

notify() 内部执行的过程:进行通知;

package thread;
 
import java.util.Scanner;
 
//创建两个线程,一个线程调用 wait,一个线程调用 notify
public class Demo18 {
    //这个对象用来作为锁对象
    public static Object locker = new Object();
 
    public static void main(String[] args) {
        Thread waitTask = new Thread(() -> {
            synchronized (locker) {
                System.out.println("wait 开始");
                try {
                    locker.wait();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("wait 结束");
            }
        });
        waitTask.start();
 
        //创建一个用来 通知/唤醒 的线程
        Thread notifyTask = new Thread(() -> {
            //让用户来控制,用户输入内容后,再执行通知~
            Scanner scanner = new Scanner(System.in);
            System.out.println("输入任意内容,开始通知:");
            //next 会阻塞,直到用户真正输入内容以后
            scanner.next();
 
            synchronized (locker) {
                System.out.println("notify 开始");
                locker.notify();
                System.out.println("notify 结束");
            }
        });
        notifyTask.start();
    }
}
 
 

运行结果:


当然,wait 和 notify 机制,还能够有效避免 "线程饿死";

线程饿死:有些情况下,调度器可能分配的不均匀,导致 有些线程反复占用 CPU,有些线程始终捞不着 CPU;

线程 在拿到锁之后,判定当下的任务是否可以进行;

如果 可以进行,那么就干活;如果不可以进行,那么就 wait;

等到合适的时候(条件满足的时候)就再继续执行(notify) / 再继续参与竞争锁;


注意:

  1. notify 在调用的时候,会尝试唤醒进行通知,如果当前对象没有在其他线程里 wait,则不会有副作用;
  2. 如果 wait 是一个对象,notify 是另一个对象,则没啥用,无法被唤醒;

3.3 notify All() 方法

 当然,在 Java 中,还有一个唤醒线程的方法 —— notifyAll() 方法;

当有多个线程等待的时候,notify 是从若干个线程里面随机挑选一个唤醒,是一次唤醒一个;而 notifyAll 则是直接唤醒所有线程,再由这些线程去竞争锁;

举个例子理解 notify 和 notifyAll 的区别:

notify 只是唤醒等待队列中的一个线程,其他的线程还是 需要乖乖的等着,如:

而 notifyAll 则是一下子将这些线程全部唤醒,这些进程则需要重新竞争锁,如: 

由于 最终的结果 notifyAll 还是只能进去一个线程,并且 其他的线程还可能出现 "线程饿死" 的情况,所以说 一般的还是 notifyAll 用的比较少 

3.4 wait 和 sleep 的对比 

都会让线程进入阻塞;

阻塞的原因和目的不同,进入的状态也不同,被唤醒的条件也不同;

wait 是用来控制线程之间的执行先后顺序,而 sleep 在实际开发中实际很少会用到(等待的时间太固定了,如果有突发情况 想提前唤醒并不是那么容易)

wait 进入的是 Waiting 状态,sleep 进入的是 Time Waiting 状态;

wait 是主动被唤醒,而 sleep 是时间到了就会自动被唤醒;

wait 其实是涵盖了 sleep 的功能,即可以死等,也可以等待最大时间,所以一般在实际开发中用的多的是 wait;


总结

本节内容我们介绍有关内存可见性和wait,notify关键字的有关知识;让我们对多线程有了进一步的学习和认识,下一节内容再见吧!!!!!!

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

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

相关文章

CUDA编程笔记(6)

文章目录前言全局内存的访问模式合并访问和非合并访问使用全局内存进行矩阵转置矩阵复制矩阵转置总结前言 全局内存的合理使用 全局内存的访问模式 合并访问和非合并访问 合并访问指的是一个线程束(同一个线程块中相邻的wrapSize个线程。现在GPU的内建变量wrapSi…

Linux系统之网络客户端工具

Linux系统之网络客户端工具一、Links工具1.Links工具介绍2.安装Links软件3.Links工具的使用4.打印网页源码输出5.打印url版本到标准格式输出二、wget工具1.wget工具介绍2.安装wget软件3.wget工具的使用三、curl工具1.curl工具的介绍2.curl的常用参数3.curl的基本使用四、scp工具…

机器学习(二)--NumPy

本篇文章介绍了一些Numpy的基础操作。NumPy 是Python语言的一个扩充程序库。支持高级大量的维度数组与矩阵运算,此外也针对数组运算提供大量的数学函数库。📙参考:NumPy 数据类型 | 菜鸟教程 (runoob.com)1.Numpy ndarray对象Numpy最重要的一…

Introduction to Multi-Armed Bandits——04 Thompson Sampling[2]

Introduction to Multi-Armed Bandits——04 Thompson Sampling[2] 参考资料 Russo D J, Van Roy B, Kazerouni A, et al. A tutorial on thompson sampling[J]. Foundations and Trends in Machine Learning, 2018, 11(1): 1-96. ts_tutorial 项目代码地址: https://githu…

蓝桥杯刷题014——求阶乘(二分法)

求阶乘 蓝桥杯2022省赛题目 问题描述 满足 N ! 的末尾恰好有 K 个 0 的最小的 N 是多少? 如果这样的 N 不存在输出 −1 。 输入格式 一个整数 K 。 输出格式 一个整数代表答案。 样例输入 2样例输出 10评测用例规模与约定 对于 30% 的数据, 1≤K≤10^6. 对于 100% 的数据, …

新瑞鹏冲刺上市:持续亏损,旗下宠物医院屡被罚,彭永鹤为董事长

家门口的宠物医院所属集团也要上市了。 1月24日,新瑞鹏宠物医疗集团有限公司(New Ruipeng Pet Group Inc.,下称“新瑞鹏”或“新瑞鹏集团”)在美国证监会(SEC)公开提交招股书,准备在美国纳斯达…

LabVIEW什么时候需要实时系统

LabVIEW什么时候需要实时系统实时计算系统能够非常可靠地执行具有非常具体时序要求的程序,这对于许多科学和工程项目来说都很重要。构建实时系统所需的关键组件是实时操作系统(RTOS)。精确计时对于许多工程师和科学家来说,在安装了…

C 语言零基础入门教程(十)

C 作用域规则 任何一种编程中,作用域是程序中定义的变量所存在的区域,超过该区域变量就不能被访问。C 语言中有三个地方可以声明变量: 1、函数或块内部的局部变量 2、在所有函数外部的全局变量 3、在形式参数的函数参数定义中 让我们来看看什…

返回值的理解

前言 我们写的函数是怎么返回的,该如何返回一个临时变量,临时变量不是出栈就销毁了吗,为什么可以传递给调用方?返回对象的大小对使用的方式有影响吗?本文将带你探究这些问题,阅读本文需要对函数栈帧有一定…

Win10+GTX3060+Python+PyTorch+Tensorflow安装

本文是个备忘录,是折腾半个下午的成果,记下来免得忘记了。 0. 安装Win10,安装显卡驱动程序。 1. 弄清楚目前版本的PyTorch和Tensorflow支持哪个版本的Python。截至本文编写时,PyTorch需要Python的3.7~3.9,Tensorflow…

【NI Multisim 14.0虚拟仪器设计——放置虚拟仪器仪表(字发生器)】

目录 序言 🍍放置虚拟仪器仪表 🍉字发生器 (1)“控件”选项组 (2)“显示”选项组 (3)“触发”选项组 (4)“频率”选项组 (5)字符…

CSS 艺术之暗系魔幻卡牌

CSS 艺术之暗系魔幻卡牌参考描述效果支线HTML图片主线去除元素的部分默认属性定义 CSS 变量body#card自定义属性定义动画#card::before#card::afterimg代码总汇参考 项目描述MDNWeb 文档搜索引擎Bing 描述 项目描述Edge109.0.1518.61 (正式版本) (64 位) 效果 注:…

DaVinci:HDR 调色

调色页面:HDR 调色Color:HDR GradeHDR 调色 HDR Grade调板不仅可用于 HDR 视频的调色, 也可用于 SDR 视频。其调色功能与标准色轮类似,但能调整的区域却要细致很多,同时,它还是可感知色彩空间的工具。高动态…

41.Isaac教程--使用DOPE进行3D物体姿态估计

使用DOPE进行3D物体姿态估计 ISAAC教程合集地址: https://blog.csdn.net/kunhe0512/category_12163211.html 深度对象姿态估计 (DOPE:Deep Object Pose Estimation) 从单个 RGB 图像执行已知对象的检测和 3D 姿态估计。 它使用深度学习方法来预测对象 3D 边界框的角点和质心的…

【数据结构】单调栈、单调队列

单调栈 单调栈 单调 栈 模拟单调递增栈的操作: 如果栈为 空 或者栈顶元素 大于 入栈元素,则入栈;否则,入栈则会破坏栈内元素的单调性,则需要将不满足条 件的栈顶元素全部弹出后,将入栈元素入栈。 单调…

研究分析如何设计高并发下的弹幕系统

一、需求背景为了更好的支持直播业务,产品设计为直播业务增加弹幕功能,但是最初的弹幕设计使用效果并不理想,经常出现卡顿、弹幕偏少等需要解决的问题。二、问题分析按照背景来分析,系统主要面临以下问题:带宽压力;弱网…

[基础]qml基础控件

TextText元素可以显示纯文本或者富文本(使用HTML标记修饰的文本)。它有font,text,color,elide,textFormat,wrapMode,horizontalAlignment,verticalAlignment等属性。主要看下clip,elide,textFormat,warpMode属性clipText 项目是可以设置宽度的…

Apache Spark 机器学习 特征抽取 4-2

Word2Vec 单词向量化是一个估算器,将文档转换成一个按照固定顺序排列的单词序列,然后,训练成一个Word2VecModel单词向量化的模型,该模型将每个单词映射成一个唯一性的、固定大小的向量集,对每个文档的所有单词进行平均…

【数据结构和算法】实现线性表中的静态、动态顺序表

本文是数据结构的开篇,上文学习了关于使用C语言实现静态、动态、文件操作的通讯录,其中使用到了结构体这一类型,实际上,是可以属于数据结构的内容,接下来我们来了解一下顺序表的相关内容。 目录 前言 一、线性表 一…

流批一体计算引擎-6-[Flink]的Python DataStream API程序

参考官方Python API文档 1 IDEA中运行Flink 从Flink 1.11版本开始, PyFlink 作业支持在 Windows 系统上运行,因此您也可以在 Windows 上开发和调试 PyFlink 作业了。 1.1 环境配置 pip3 install apache-flink1.15.3 CMD>set PATH查看环境变量 CMD>set JAV…