【JavaEE初阶】多线程(三)volatile wait notify关键字 单例模式

news2024/11/15 14:03:09

摄影分享~~
在这里插入图片描述

文章目录

  • volatile关键字
    • volatile能保证内存可见性
  • wait和notify
    • wait
    • notify
    • notifyAll
    • wait和sleep的区别
    • 小练习
  • 多线程案例
    • 单例模式
      • 饿汉模式
      • 懒汉模式

volatile关键字

volatile能保证内存可见性

import java.util.Scanner;

class MyCounter {
    public int flag = 0;
}

public class ThreadDemo14 {
    public static void main(String[] args) {
        MyCounter myCounter = new MyCounter();

        Thread t1 = new Thread(() -> {
            while (myCounter.flag == 0) {
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            System.out.println("t1 循环结束");
        });

        Thread t2 = new Thread(() -> {
            Scanner scanner = new Scanner(System.in);
            System.out.println("请输入一个整数: ");
            myCounter.flag = scanner.nextInt();
        });

        t1.start();
        t2.start();
    }
}

以上代码运行的结果可能是输入1后,t1这个线程并没有结束。而是一直在while中循环。而t2线程已经执行完了。
以上情况,就叫做内存可见性问题
在这里插入图片描述
这里使用汇编来理解,大概分为两步操作:

  1. load,把内存中flag的值,读到寄存器中。
  2. cmp,把寄存器中的值,和0进行比较。根据比较结果,决定下一步往哪个地方执行(条件跳转指令)

上述循环循环体为空,循环执行速度极快。循环执行很多次,在t2真正修改之前,load得到的结果都是一样的。另一方面,load操作和cmp操作相比,速度慢的多得多。由于load执行速度太慢(相比于cmp),再加上反复load到的结果都是一样的,JVM就做出了一个大胆的决定:不再真正的重复load,判定没有人修改flag值(但实际上是有人在修改的,t2在修改),直接就读取一次就好。(编译器优化的一种方式)
内存可见性问题:一个线程针对一个变量进行读取操作,同时另一个线程针对这个变量进行修改。此时读到的值,并不一定是修改之后的值。(jvm/编译器在多线程环境下优化时残生了误判)
此时,我们就需要手动干预了。我们可以给flag这个变量加上volatile关键字。告诉编译器,这个变量是“易变”的,需要每一次都重新读取这个变量的内容。
在这里插入图片描述
volatile不保证原子性,原子性是由synchronized来保证的。

wait和notify

举个列子:
t1,t2两个线程,希望t1先执行任务,任务执行快结束了让t2来干,就可以让t2先wait(阻塞,主动放弃cpu)。等t1任务执行快结束了,在通过notify通知t2,把t2唤醒,让t2开始执行任务。
上述场景中,使用join和sleep可以吗?
使用join,必须要t1彻底执行完,t2才能执行。如果希望t1执行一半任务然后让t2执行,join无法完成。
使用sleep,必须制定一个休眠时间,但是t1执行任务的时间是难以估计的。
使用wait和notify可以解决上述问题。

wait

wait进行阻塞,某个线程调用wait方法,就会进入阻塞,此时就处于WAITING.
在这里插入图片描述

这个异常,很多带有阻塞功能的方法都带,这些方法都是可以被interrupt方法通过以上异常唤醒。
我们再来看一个代码:

public class ThreadDemo17 {
        public static void main(String[] args) throws InterruptedException {
            Thread t = new Thread(() -> {
                try {
                    Thread.sleep(5000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("执行完毕!");
            });

            t.start();
            System.out.println("wait前");
            t.wait();
            System.out.println("wait后");
        }
}

在这里插入图片描述
这里会出现非法的锁状态异常。锁的状态一般是被加锁的状态,被解锁的状态。
为什么会出现这个异常呢?和wait的操作有关:

wait的操作:

  1. 先释放锁
  2. 进行阻塞等待
  3. 收到通知后,重新尝试获取锁,并且在获取锁后,继续往下执行。

上述代码没有锁就想要释放锁,所以出现了非法的锁状态异常。
因此,wait操作要搭配synchronized来使用。
在这里插入图片描述

notify

wait和notify一般搭配使用。notify方法用来唤醒wait等待的线程, wait能够释放锁, 使线程等待, 而notify唤醒线程后能够获取锁, 然后使线程继续执行。
在这里插入图片描述
如果上述代码中,t1还没有执行wait,t2已经执行了notify,那么此时的声明就是没有用的。t2执行notify后,t1执行wait后会一直阻塞等待。
注意上述代码在t2唤醒t1之后,t1和t2之间的执行是随机的,也是就标号3和标号4的地方的顺序是不确定的。

方法效果
wait();无参数,一直等直到notify唤醒
wait(时间参数);指定最长等待时间

notifyAll

notify方法只是唤醒某一个等待线程. 使用notifyAll方法可以一次唤醒所有的等待线程.
一般情况下,使用notify。因为全部唤醒会导致线程之间抢占式执行。不一定安全。

wait和sleep的区别

相同点:

  • 都可以使线程暂停一段时间来控制线程之间的执行顺序.
  • wait可以设置一个最长等待时间, 和sleep一样都可以提前唤醒.

不同点:

  • wait是Object类中的一个方法, sleep是Thread类中的一个方法.
  • wait必须在synchronized修饰的代码块或方法中使用, sleep方法可以在任何位置使用.
  • wait被调用后当前线程进入BLOCK状态并释放锁,并可以通过notify和notifyAll方法进行唤醒;sleep被调用后当前线程进入TIMED_WAITING状态,不涉及锁相关的操作.
  • 使用sleep只能指定一个固定的休眠时间, 线程中执行操作的执行时间是无法确定的; 而使用wait在指定操作位置就可以唤醒线程.
  • sleep和wait都可以被提前唤醒, interruppt唤醒sleep, 是会报异常的, 这种方式是一个非正常的执行逻辑; 而noitify唤醒wait是正常的业务执行逻辑, 不会有任何异常.

小练习

有三个线程, 分别只能打印 A, B, C. 控制三个线程固定按照 ABC 的顺序来打印.

public class ThreadDemo18 {
    // 有三个线程, 分别只能打印 A, B, C. 控制三个线程固定按照 ABC 的顺序来打印.
    public static void main(String[] args) throws InterruptedException {
        Object locker1 = new Object();
        Object locker2 = new Object();
        Thread t1 = new Thread(()->{
            System.out.println("A");
            synchronized (locker1) {
                locker1.notify();
            }
        });
        Thread t2 = new Thread(()->{
            synchronized (locker1) {
                try {
                    locker1.wait();
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }

            }
            System.out.println("B");
            synchronized (locker2) {
                    locker2.notify();
            }
        });
        Thread t3 = new Thread(()->{
            synchronized (locker2) {
                try {
                    locker2.wait();
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }
            System.out.println("C");
        });
        t2.start();
        t3.start();
        Thread.sleep(100);
        t1.start();
    }
}

创建locker1,供1,2使用
创建locker2,供2,3使用
线程3,locker2.wait()
线程2, locker1.wait()唤醒后执行locker2.notify
线程1执行自己的任务,执行完后locker.notify
在这里插入图片描述

多线程案例

单例模式

单例模式是设计模式的一种。
单例模式能保证某个类在程序中只存在唯一一份的实例,而不会创建出多个实例。
单例模式具体的实现方式分为“饿汉”和“懒汉”。

饿汉模式

类加载的同时,创建实例。
类对象在一个java进程中,只有一份。因此类对象内部的类属性也是唯一的。
在类加载阶段,就把实例创建出来了。

//饿汉模式的单例模式的实现
//保证Singleton这个类只能创建出一个实例
class Singleton{
    //在此处,先将实例创建出来
    private static Singleton instance = new Singleton();

    public static Singleton getInstance() {
        return instance;
    }
    //为了避免Singleton类不小心被多复制出来
    //把构造方法设为private,在类外,无法通过new的方式来创建一个Singleton
    private Singleton(){

    }
}
public class ThreadDemo19 {
    public static void main(String[] args) {
        Singleton s = Singleton.getInstance();
        Singleton s2 = Singleton.getInstance();
        //Singleton s3 = new Singleton();
        System.out.println(s == s2);
    }
}

在这里插入图片描述
在这里插入图片描述

  • static保证这个实例唯一
  • static保证这个实例被创建出来。

懒汉模式

class SingletonLazy{
    private static SingletonLazy instance = null;


    public static SingletonLazy getIsntance() {
        if(instance == null){
            instance = new SingletonLazy();
        }
        return instance;
    }
}
public class ThreadDemo20 {
        public static void main(String[] args) {
            SingletonLazy s = SingletonLazy.getIsntance();
            SingletonLazy s2 = SingletonLazy.getIsntance();
            System.out.println(s == s2);
        }
}

在这里插入图片描述

在多线程中调用instance,饿汉模式是线程不安全的。
在这里插入图片描述
那么如何保证懒汉模式线程安全呢?
**加锁。**线程安全的本质问题,就是读,比较,写这三个操作不是原子的。所以我们可以加锁来解决线程安全问题。
在这里插入图片描述

但是,加锁操作就导致每次调用getInstance都需要花一定的开销。而我们的加锁只针对new对象之前,所以我们就可以判断一下对象是否创建,再去决定加锁。
如果对象创建了,就不加锁。如果对象没有创建,就加锁。
在这里插入图片描述
上述代码还存在一个问题,即内存可见性问题:
假如调用getInstance的线程有很多,此时代码就有可能被优化(第一次读内存,后续读的是寄存器/cache)
除此之外,可能还会涉及到指令重排序。
在这里插入图片描述
上述代码中,分为三个步骤:

  1. 申请内存空间
  2. 调用构造方法,把这个内存空间初始化成一个对象
  3. 把内存空间的地址赋值给instance引用

而编译器的指令重排序操作就会调整代码执行顺序,123可能会变成132.(单线程中没有影响)
我们可以给代码中加上volatile。
volatile有两个功能:

  1. 解决内存可见性
  2. 禁止指令重排序。

以下为懒汉模式的单例模式的完整代码:

class SingletonLazy{
    private volatile static SingletonLazy instance = null;


    public static SingletonLazy getInstance() {
        if(instance ==null){
            synchronized (SingletonLazy.class){
                if(instance == null){
                    instance = new SingletonLazy();
                }
            }
        }  

        return instance;
    }
}
public class ThreadDemo20 {
        public static void main(String[] args) {
            SingletonLazy s = SingletonLazy.getInstance();
            SingletonLazy s2 = SingletonLazy.getInstance();
            System.out.println(s == s2);
        }
}

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

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

相关文章

2023新型智慧城市解决方案(ppt可编辑)

本资料来源公开网络,仅供个人学习,请勿商用,如有侵权请联系删除 智慧城市建设整体框架 智慧城市建设内容 智慧城市云基础架构的必要性 智慧城市建设效益 智慧城市建设效益17 智慧城市建设模式建议 智慧城市建设-整体解决法方案 智慧城市建设…

关于Nginx

一、常见的“服务器中间件”(即http server-web中间件)有哪些 Tomcat、Jboss、Apache、WeBlogic、Jetty、webSphere、Nginx、IIS 二、nginx的特点 1.性能高,能承受5万并发每秒; 2.内存、磁盘,读取消耗空间小。 三、…

「背包问题-步入」失衡天平

失衡天平 https://ac.nowcoder.com/acm/contest/24213/1021 题目描述 终于Alice走出了大魔王的陷阱,可是现在傻傻的她忘了带武器了,这可如何是好???这个时候,一个神秘老人走到她面前答应无偿给她武器,但老人有个条件&#xff…

2023-04-24 算法面试中常见的贪心算法问题

贪心算法 1 贪心选择例题 455.饼干分配 假设你想给小朋友们饼干。每个小朋友最多能够给一块儿饼干。每个小朋友都有一个“贪心指数”,称为g(i),g(i)表示的是这名小朋友需要的饼干大小的最小值。同时,每个饼干都有一个大小值s(i)。如果s(j)…

转换json格式的日期为Javascript对象的函数

项目中碰到了用jQuery从后台获取的json格式的日期的字符串,需要将此字符串转换成JavaScript的日期对象。 代码如下: //转换json格式的日期(如:{ServerDatetime:"\/Date(1278930470649)\/"})为Javascript的日期对象 fu…

【离散系统】传递函数和状态空间方程离散化

本文如有错误,恳请指正。 目录 离散系统 采样控制系统 数字控制系统 信号采样 采样定理(香农定理) 信号保持—零阶保持器 Z变换 Z 变换方法 级数求和法 部分分式法 基本定理 Z反变换 Z反变换方法 长除法 部分分式法&#xff0…

hook函数,toRef家族

自定义hook函数 什么是hook?——本质是一个函数,把setup函数中使用的组合式API 进行了封装。 类似于vue2.x中的mixin。 自定义hook的优势:复用代码,让setup中的逻辑更清楚易懂。 定义钩子: 使用钩子 toRef,toRefs 作用:创建一个ref对象…

WinScope实现录制视频与是Timeline时间轴同步设置方法-千里马framework车载手机系统开发实战

hi,粉丝朋友们! 背景: 今天来分享一个粉丝朋友提出的问题,那就是他在学习wms课程时候有用到winscope工具,提出一个疑问,就是google官网说的有录屏可以结合起来一起看。具体如下: 其实这个以…

Vue 手搓轮播效果

tiptop: 为啥需要写这个功能,因为我遇到了每个轮播层内要放3个左右的商品块,如果使用element自带的轮播就需要将一维数组切成二维数组,导致处理一些情况下就会变得很麻烦,当然那种我也写了如果你们有需要,在下方留言我…

柳永8首绝美词句

他是白衣卿相,是才子词人。 他有傲人的才华,却仕途失意。 他眠花宿柳,写下流传千古的词章。 他是才子柳永。 今天,给大家介绍他的8首词,体会柳永的人生起伏。 1、《雨霖铃 寒蝉凄切》 寒蝉凄切,对长亭晚…

Kafka 安装部署-单节点

Kafka强依赖ZK,如果想要使用Kafka,就必须安装ZK,Kafka中的消费偏置信息、kafka集群、topic信息会被存储在ZK中。有人可能会说我在使用Kafka的时候就没有安装ZK,那是因为Kafka内置了一个ZK,一般我们不使用它。 部署说明…

Yuzuki Lizard 全志V851S开发板 –移植 QT5.12.9教程

本文转载自:https://forums.100ask.net/t/topic/3027 移植 QT5 教程 (此教程基于docker版V851S开发环境) docker pull registry.cn-hangzhou.aliyuncs.com/gloomyghost/yuzukilizard 编译依赖 apt-get install repo git gcc-arm-linux-gn…

用户订阅付费如何拆解分析?看这篇就够了

会员制的订阅付费在影音娱乐行业中已相当普及,近几年,不少游戏厂商也开始尝试订阅收费模式。在分析具体的用户订阅偏好以及订阅付费模式带来的增长效果时,我们常常会有这些疑问: 如何从用户的整体付费行为中具体拆解订阅付费事件…

Numpy从入门到精通——数组变形|合并数组

这个专栏名为《Numpy从入门到精通》,顾名思义,是记录自己学习numpy的学习过程,也方便自己之后复盘!为深度学习的进一步学习奠定基础!希望能给大家带来帮助,爱睡觉的咋祝您生活愉快! 这一篇介绍《…

SLAM论文速递:SLAM—— 流融合:基于光流的动态稠密RGB-D SLAM—4.25(2)

论文信息 题目: FlowFusion:Dynamic Dense RGB-D SLAM Based on Optical Flow 流融合:基于光流的动态稠密RGB-D SLAM论文地址: https://arxiv.org/pdf/2003.05102.pdf发表期刊: 2020 IEEE International Conference on Robotics and Automa…

【Spring Cloud】Spring Cloud 是什么?

文章目录 前言一、子项目二、常用组件三、把 Spring Cloud 官方、Netflix、Alibaba 三者整理成如下表格: 前言 Spring 以 Bean(对象) 为中心,提供 IOC、AOP 等功能。Spring Boot 以 Application(应用) 为中…

Python小姿势 - Python操作Excel表格数据

Python操作Excel表格数据 在日常工作中,我们经常会遇到一些需要对表格数据进行处理的情况,比如:数据清洗、数据分析、数据可视化等。对于这些任务,我们可以使用Python来进行操作。 在Python中操作Excel表格数据,我们可…

electron+vue3全家桶+vite项目搭建【13】封装加载进度显示,新建窗口 演示主进程与渲染进程通信

文章目录 引入实现效果演示:1.封装新建窗口工具2.测试新建窗口3.封装进度条加载4.测试进度条加载 引入 这里我们通过封装electron的工具类来演示electron中的主进程和渲染进程利用ipc进行通信 demo项目地址 electron官方文档ipc通信 实现效果演示: …

通过注册表设置远程桌面的配置

***************************修改远程终端端口号*************************** [HKEY_LOCAL_MACHINE\SYSTEM\ControlSet001\Control\Terminal Server\Wds\rdpwd\Tds\tcp] "PortNumber"dword:00000d3d [HKEY_LOCAL_MACHINE\SYSTEM\ControlSet001\Control\Terminal Ser…

代码优雅之道——如何干掉过多的if else

1、前言 注意标题是过多的,所以三四个就没必要干掉了。实际开发中我们经常遇到判断条件很多的情况,比如下图有20多种情况,不用想肯定是要优化代码的,需要思考的是如何去优化? 网上很多说用switch case啊,首…