线程调度与单例模式:wait、notify与懒汉模式解析

一.wait 和 notify(等待 和 通知)

引入 wait notify 就是为了能够从应用层面,干预到多个不同线程代码的执行顺序,可以让后执行的线程主动放弃被调度的机会,等先执行的线程完成后通知放弃调度的线程重新执行。

自助取款机

当取款机没有钱的时候,要想去取钱只能等别人进去存钱或者等银行的人过来放钱,否则他永远拿不到钱,他在出去之后又会进去取款,刚刚释放了锁,就又会参与到锁竞争,并且大概率他会一直拿到锁,这叫“线程饿死“,这种情况只是概率性事件,但还是会极大影响到其他线程运行,这个时候他为了不影响到其他人就会等待(wait),然后让后面如果存钱的人存了钱了,通知(notify)自己一声,自己就可以重新排队取钱,在没通知的这段时间,他就会一直在旁边等,不会去排队了。 

join 和 wait

join 是等待另一个线程执行完,才继续执行
wait 则是等待另一个线程通过 notify 进行通知(不要求另一个线程必须执行完)
wait进入阻塞,只能说明自己释放锁了,至于是否有其他线程拿到锁,这是不能确定的
public class ThreadDemo25 {public static void main(String[] args) {//需要有一个统一的对象进行加锁,wait,nitifyObject locker = new Object();Thread t1 = new Thread(() -> {synchronized (locker){System.out.println("t1 wait 之前");try {locker.wait();} catch (InterruptedException e) {throw new RuntimeException(e);}System.out.println(" t1 wait 之后");}});Thread t2 = new Thread(() -> {try {Thread.sleep(5000);synchronized (locker){System.out.println(" t2 notify 之前");locker.notify();System.out.println("t2 notify 之后");}} catch (InterruptedException e) {throw new RuntimeException(e);}});t1.start();t2.start();}
}

我们发现阻塞等待的原因是19行的wait

wait方法放到synchronized是因为要释放锁,前提是先加上锁

java特别约定要把notify放到synchronized中 

由于线程的随机调度我们并不知道要先调用谁,如果先调用t2就没有线程去给t1唤醒了,所以要保证t1先执行,我们给t2加上了sleep 

需要注意的是t2在notify之后并没有释放锁,而t1唤醒后会尝试加锁,所以会产生小小的阻塞。

notifyAll     唤醒这个对象所有等待的线程

假设有很多个线程,都使用同一个对象wait,针对这个对象进行notifyAll,此时就会全都唤醒

需要注意的是,这些线程在wait返回的时候,需要重新获取锁,就会因为锁竞争,是这些线程串行执行

wait 和 sleep 

wait 提供了一个带有超时时间的版本
sleep 也是能指定时间
都是时间到了,就继续执行,解除阻塞了
wait 和 sleep 都可以被提前唤醒(虽然时间还没到),wait通过notify唤醒,sleep通过interrupt唤醒
使用wait 最主要的目标一定是不知道多少时间的前提下使用的,超时时间是为了兜底
使用sleep,一定是知道了多少时间的前提下使用的,虽然能提前唤醒,但是通过异常唤醒一般是程序出现一些特殊情况了

二.单例模式

单例模式是一种设计模式,遵守设计模式下限就有了兜底

1.饿汉模式

单例 == 单个实例(对象)

static 这个引用就是我们期望创建出唯一的实例的引用,static 静态的 指的是“类属性”

instance 就是 当前类对象里面持有的属性

每个类的类对象,只存在一个,类对象中的static属性,自然也是只有一个了

 这时instance所指向的对象就是唯一的一个对象,其他代码要想使用这个类的实例,就需要通过这个方法来进行获取,而不是直接new一个出来。

                                 

这直接从根本上让其他人想new都new不了了

Sington内部代码早就把唯一的实例安排好了 

class Singleton{private static Singleton instance = new Singleton();public static Singleton getInstance(){return instance;}private Singleton() {}
}
public class ThreadDemo26 {public static void main(String[] args) {Singleton s = Singleton.getInstance();}
}

上述代码成为“饿汉模式”单例模式一种简单的写法,在程序启动时,实例就创建了,所以就是用饿汉,创建实例非常早。

 2.懒汉模式

创建实例的时机不一样了,创建实例的时机更晚,直到第一次使用的时候,才会创建实例

class SingletonLazy{//这个引用指向唯一实例,这个引用先初始化null,而不是立即创建实例private volatile static SingletonLazy instance = null;private static Object locker = new Object();public static SingletonLazy getInstance(){if(instance == null){//如果 Instance 为 NULL,就说明时首次调用,首次调用就需要考虑线程安全问题,就要加锁//如果非空的话就说明时后续调用就不必加锁synchronized (locker){if(instance == null) instance = new SingletonLazy();}}return instance;}private SingletonLazy(){}
}
public class ThreadDemo27 {public static void main(String[] args) {SingletonLazy s1 = SingletonLazy.getInstance();SingletonLazy s2 = SingletonLazy.getInstance();System.out.println(s1 == s2);}
}

如果是首次调用 getinstance,那么instance 引用 为null,进入if语句创建实例出来,后续再次调用返回的就已经是创建好的引用了。

饿汉模式和懒汉模式是否属于线程安全?

 饿汉模式属于读操作在多线程是安全的。
但加入了懒汉模式就不一定了

 这样就会导致实例被new 了两次,就又bug了。这时线程不安全,在多线程可能会new出多个实例

即使写成这样没有线程安全问题,但还是因为已经创建了实例,但还是进行加锁解锁操作使得效率降低。

 所以if不一定要加在方法上,

我们加在了实例化对象上,这样就不会木讷的进行加锁,但是还有一个致命的问题就是整个方法不是原子了,这个时候我们就得考虑对象是否创建问题了。

我们得先理清一下思路了,现在我们第一个if是用来判断是不是第一次进行创建对象的,我们所面临的问题是在多线程下if后面的代码执行是不确定的,可能已经调用其他线程并创建了对象,所以我们得加一个判断是否创建了对象的语句

 两个if代码一样但意义不同

第一个if是为了判定是否第一次创建对象,并加锁,第二个if是判断是否要创建对象

在创建对象过程中还涉及到了线程安全问题 ------- 指令重排序

调整原有代码的执行顺序,保证逻辑不管的前提下,提高程序的效率

创建对象这一行代码可以拆为三个步骤

1.申请一段内存空间
2.在这个内存上调用的方法。创建出这个实例
3.把这个内存地址赋值给instance引用变量

正常情况下是按照1,2,3顺序来执行的,但是编译器可能优化成1,3,2的顺序来执行

在单线程下1,2,3或者1,3,2都是可以的

但是如果是多线程就可能引入问题了!!!

在t1加锁之后t2进行阻塞,t1解锁后t2获得了锁,但在这个时候 t2判断不为空直接返回,但这个时候instance并未初始化,如果使用instance里面的属性或者方法就可能会出现错误,那难道2不是同时进行的吗?
注意:在执行完1,3后线程有可能也被调度走了,并未进行初始化。要想执行2可能会隔一段时间

在之前解决内存可见性时我们用到了volatile它的功能有两个

1.保证内存可见性,每次访问变量必须都要重新读取内存,而不会优化到寄存器/缓存中
2.禁止指令重排序,针对这个volatile 修饰的变量的读写操作相关指令,是不能被重排序的

回顾一些我们解决问题的步骤

首先

我们因为饿汉模式在多线程下,会出现二次实例化对象的操作,所以我们加上了锁,

第二次

我们因为即使加上了锁,但我们因为加锁的位置太消耗效率所以我们将锁的位置改变了,但我们无法判断对象是否被创建了,所以我们又加上了一层if,

第三次

我们了解了指令重排序,我们会遇到,对象创建但并未初始化,然后导致使用了没初始化对象的属性或者方法,出现了失误,这时候我们回想起之前解决内存可见性的volatile,它的另一个功能就是解决指令重排序所以我们加上了这个关键字,至此我们解决了这一系列问题

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

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

相关文章

ros运行包,Ubuntu20.04成功运行LIO-SAM

zz:~/lio_sam_ws$ source devel/setup.bash zz:~/lio_sam_ws$ roslaunch lio_sam run.launch 创建包链接: 链接1:Ubuntu20.04成功运行LIO-SAM_ubuntu20.04运行liosam-CSDN博客 链接2:ubuntu 20.04 ROS 编译和运行 lio-sam,并且导出PCD文件…

AI自动化工作流:开启当下智能生产力的价值

举手之言:AI自动化工作流创造了什么呢? AI自动化工作流 ,顾名思义,是将人工智能(AI)技术与自动化流程相结合,通过智能化的方式来完成复杂的任务和操作。简单来说,它就是利用AI的强大…

【设计模式】- 行为型模式2

观察者模式 定义了一对多的依赖关系,让多个观察者对象同时监听某一个对象主题。这个主题对象在状态变化时,会通知所有的观察者对象,让他们能够自动更新自己。 【主要角色】 抽象主题角色:把所有观察者对象保存在一个集合里&…

mapbox-gl强制请求需要accessToken的问题

vue引入"mapbox-gl": "^2.15.0", 1.13以后得版本,都强制需要验证这个mapboxgl.accessToken。 解决办法:实例化地图的代码中,加入这个: const originalFetch window.fetch; window.fetch function ({ url…

已知6、7、8月月平均气温和标准差,求夏季季平均温度与标准差

由下面定理,得出平方和的公式:(即每天的温度平方和) 这样就可以推出季平均的算法: 举例:在Excel用公式算,不要手算: 因此季平均:(B2*C2B3*C3B4*C4)/SUM(B2:B4) 季标准差…

手机内存不够,哪些文件可以删?

1️⃣应用缓存文件 安卓:通过「文件管理器」→「Android」→「data」或「cache」文件夹(部分需权限),或直接在应用设置中清除缓存 iOS:无需手动清理,系统会自动管理,或在应用内设置中清除&…

可编辑98页PPT | 某大型制造业数字化转型战略规划项目方案

荐言摘要:某大型制造业数字化转型战略规划项目方案聚焦企业全价值链升级,以“数据驱动业务重塑”为核心,打造行业标杆级数字化能力。项目将分三阶段推进,首阶段聚焦顶层设计,通过现状诊断明确痛点:针对企业…

lovart design 设计类agent的系统提示词解读

文章目录 lovart 设计agent介绍角色定义工作规范工具调用任务复杂度指南任务移交指南其他ref lovart 设计agent介绍 lovart作为设计agent,产品功能包括: 全链路设计能力:可以快速生成完整的品牌视觉方案,包括标志、配色、品牌规范…

使用 docker-volume-backup 备份 Docker 卷

docker-volume-backup 是一个用于备份 Docker 卷的工具,在 Windows 10 上使用它,你可以按照以下步骤操作: 1. 确保 Docker 环境已安装并正常运行 在 Windows 10 上,你需要安装 Docker Desktop for Windows。可以从 Docker 官方网…

用户行为日志分析的常用架构

## 1. 经典Lambda架构 Lambda架构是一种流行的大数据处理架构,特别适合用户行为日志分析场景。 ### 1.1 架构组成 Lambda架构包含三层: - **批处理层(Batch Layer)**: 存储全量数据并进行离线批处理 - **实时处理层(Speed Layer)**: 处理最新数据&…

从API到UI:直播美颜SDK中的滤镜与贴纸功能开发与落地方案详解

时下,滤镜和贴纸功能,已经成为主播们展现个性、增强互动的“必备神器”。那么,这些功能背后的技术实现到底有多复杂?如何从API到UI构建一个流畅、灵活的美颜SDK呢?本文将从底层原理到前端实现,全面解析这两…

21.EC实战 嵌入式控制器EC如何进入休眠模式实现低功耗

文章目录 一、概述1. WUI0中断向量表配置2. 中断服务函数内容3. 深度睡眠检测4. 深度睡眠功能函数4.1 关闭所有中断4.2 外部中断对应引脚功能配置4.3 设置唤醒功能和唤醒中断4.4 进入深度睡眠状态一、概述 EC作为笔记本电脑的嵌入式控制器,在笔记本电脑使用电池单独工作时,关…

Java实现PDF加水印功能:技术解析与实践指南

Java实现PDF加水印功能:技术解析与实践指南 在当今数字化办公环境中,PDF文件因其跨平台兼容性和格式稳定性而被广泛应用。然而,为了保护文档的版权、标记文档状态(如“草稿”“机密”等)或增加文档的可追溯性&#xf…

vue2、vue3项目打包生成txt文件-自动记录打包日期:git版本、当前分支、提交人姓名、提交日期、提交描述等信息 和 前端项目的版本号json文件

vue2 打包生成text文件 和 前端项目的版本号json文件 项目打包生成txt文件-自动记录git版本、当前分支、提交人姓名、提交日期、提交描述等信息生成版本号json文件-自动记录当前版本号、打包时间等信息新建branch-version-webpack-plugin.js文件 // 同步子进程 const execSyn…

Filament引擎(一) ——渲染框架设计

filament是谷歌开源的一个基于物理渲染(PBR)的轻量级、高性能的实时渲染框架,其框架架构设计并不复杂,后端RHI的设计也比较简单。重点其实在于项目中材质、光照模型背后的方程式和理论,以及对它们的实现。相关的信息,可以参考官方…

洛谷B3876—— [信息与未来 2015] 中间值

见:B3876 [信息与未来 2015] 中间值 - 洛谷 题目描述 给出一个正整数 n,生成长度为 n 的数列 a,其中 ai​i(1≤i≤n)。 若 n 为奇数,则输出 a 的中间数(位于 a 正中位置的数);若 n 为偶数&am…

Java 后端基础 Maven

Maven 1.什么是Maven 2.Maven的作用 Maven核心 Maven概述 IDEA集成Maven 1.创建Maven项目 点击设置里的 Project Structure 将jdk和编译语言进行设置 随后点击apply点击ok 2.Maven坐标 3.导入Maven项目 将文件夹复制到当前项目的目录下 在这个目录下,在磁盘中…

qtcreater配置opencv

我配置opencv不管是按照网上的教程还是deep seek发现都有些问题,下面是我的配置方法以及实践成功的心得 电脑环境 windows平台qt6 下载 我这里直接提供官网下载地址:https://opencv.org/releases/ 我下载的是最新版,下载后是一个.exe文件…

单片机-STM32部分:15、直流电机与步进电机 PWM/IO

飞书文档https://x509p6c8to.feishu.cn/wiki/InUfwEeJNimqctkyW1mcImianLh 一、步进电机与直流电机: 1-1、什么是直流电机? 直流电机是最常见的电机类型。直流电动机通常只有两个引线,一个正极和一个负极。直流电机的转速控制主要依靠改变输…

「佰傲再生医学」携手企企通,解锁企业采购供应链数字化新体验

健康,是人类美好生活的基石。随着“健康中国2030”规划的深入推进,生物医药和再生医学等前沿技术快速崛起,已成为促进全民健康、提升生命质量的重要支撑,为健康事业注入了新的希望和动力。 一、佰傲再生医学,让每个人…