wait 和 notify
目录
wait() 方法
notify() 方法
notifyAll() 方法
nofity 和 notifyAll
wait 和 notify
wait 和 sleep 的区别
wait 和 join 的区别
由于线程之间是抢占式执行的,因此,线程之间执行的先后顺序难以预知,但是,在实际开发中,有时候我们希望合理的协调多个线程之间的执行先后顺序
例如:
在篮球场上,每个队员都是独立的 执行流,也就是一个 线程
当需要完成一个具体的得分动作时,就需要多个队员相互配合,按照一定的顺序执行一定的动作,线程 1 先向 线程 2 "传球",线程2 才能 "扣篮"
要完成 协调工作,主要涉及到三个方法:
wait()/wait(long timeout):让当前线程进入等待状态
notify():唤醒当前对象上等待的线程
notifyAll():唤醒当前对象上所有等待的线程
接下来,我们就来学习这三个方法,我们首先来看 wait 方法
wait() 方法
我们先来看一个例子:
一个柜子里有食物,5个人(线程)共同使用这个柜子,并从里面拿取食物(假设同一时间只能一人拿取),1号先使用这个柜子(对其进行加锁),但是当1号打开这个柜子时发现柜子里没有食物,此时1号就会关上柜门(释放锁),等有食物时再来拿
此时,其他人就会竞争这个锁,争取使用柜子,而刚刚释放锁的1号,也会参与到锁竞争中,因此,也就有可能刚刚释放锁的1号又重新拿到锁,并且,由于1号离得近(1号线程处于 RUNNABEL 状态,其他线程处于 BLOCKED 状态),他就有很大可能再次拿到锁
1号又拿到锁,发现没有食物,又释放锁,又竞争到锁,发现没有食物,又释放锁......
如此重复,就会导致1号反复获取到锁,但是又不能完成实质性的操作;而其他线程,则无法拿到锁。这种情况,称之为 线程饿死(线程饥饿)
线程饿死这样的情况,属于概率性事件(1号拿到锁的概率更大,但是其他线程也有可能会拿到锁),不像 死锁,一旦出现后,就一定会阻塞,但 线程饥饿 这样的情况,也极大可能会影响其他线程的运行
因此,我们就需要对这种情况进行处理:
线程饿死出现的关键在于:1号发现自己要执行逻辑的前提条件不具备(柜子中没有食物)时,就应该主动放弃对锁的竞争,主动放弃去 CPU 上调度执行,即,进入阻塞状态,一直等待前提条件具备了(其他线程往柜子中放了食物),此时再解除阻塞,参与到锁竞争中
此时,就可以使用 wait 进行等待:
让 1号 判断前提条件是否满足,若不满足,则 wait 等待
其他线程让条件满足后,再通过 notify 来唤醒 1号
接下来,我们就通过具体的代码来学习 wait 的使用:
我们让 t1 线程进入等待:
public class ThreadDemo18 {public static void main(String[] args) {Object locker = new Object();Thread t1 = new Thread(() -> {System.out.println("t1 进入等待之前");// 进入等待try {locker.wait();} catch (InterruptedException e) {e.printStackTrace();}});t1.start();}
}
此时,观察运行结果,发现抛出了异常 IllegalMonitorStateException

为什么会抛出异常呢?
这是因为 wait 必须搭配 synchronized 来使用
wait 做的事情有:
1. 使当前执行代码的线程进行等待(将线程放到等待队列中)
2. 释放当前锁
3. 满足一定条件时被唤醒,重新尝试获取到这个锁
wait 要对当前锁进行释放,释放锁的前提,是要先拿到锁,因此 wait 必须放到 synchronized 中使用
public class ThreadDemo18 {public static void main(String[] args) {Object locker = new Object();Thread t1 = new Thread(() -> {System.out.println("t1 进入等待之前");synchronized (locker) {// 进入等待try {locker.wait();} catch (InterruptedException e) {e.printStackTrace();}}});t1.start();}
}
wait、sleep 和 join,都有可能被 interrupt 提前唤醒,都需要处理异常
每个对象里都有一把锁,调用 wait 的对象,必须和 synchronized 中的锁对象是一致的,wait 解除的锁是 locker对象 的锁,后续 wait 被唤醒后,重新获取到锁,当然也是获取到 locker 对象的锁
此时,t1 线程就在 wait 这里阻塞了:

我们使用 jconsole 来观察 t1 线程的状态:

此时,线程进入 WAITING 状态
而 wait 结束等待的条件为:
1. 其他线程调用该对象的 notify 方法
2. wait 等待时间超时(wait 方法提供了带有 timeout 参数的版本,用来指定等待最长时间)
3. 其他线程调用该等待线程的 interrupt 方法,导致 wait 抛出 InterruptedException 异常
接下来,我们就来学习 notify() 方法,来唤醒等待中的线程
notify() 方法
notify() 方法用于唤醒等待的线程
我们在 t2 线程中唤醒 t1 线程:
public class ThreadDemo18 {public static void main(String[] args) {Object locker = new Object();Thread t1 = new Thread(() -> {System.out.println("t1 进入等待之前");synchronized (locker) {// 进入等待try {locker.wait();} catch (InterruptedException e) {e.printStackTrace();}}});Thread t2 = new Thread(() -> {try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}locker.notify();});t1.start();t2.start();}
}
此时抛出异常 IllegalMonitorStateException

这是因为 Java 中约定 notify 也需要放到 synchronized 中
public class ThreadDemo18 {public static void main(String[] args) {Object locker = new Object();Thread t1 = new Thread(() -> {System.out.println("t1 进入等待之前");synchronized (locker) {// 进入等待try {locker.wait();} catch (InterruptedException e) {e.printStackTrace();}}System.out.println("t1 结束等待");});Thread t2 = new Thread(() -> {try {Thread.sleep(1000);synchronized (locker) {System.out.println("t2 notify 之前");locker.notify();System.out.println("t2 notify 之后");}} catch (InterruptedException e) {e.printStackTrace();}});t1.start();t2.start();}
}

在上述代码中:
t1 执行起来后就会立即尝试获取锁,拿到锁后,就立即打印 "t1 进入等待之前",并进入 wait 方法(释放锁且阻塞等待)
t2 执行起来后,会先 sleep(1000)(保证 t1 能够先拿到锁)
t2 sleep 之后,t1 处于 WAINTING 状态,且锁是释放了的,此时,t2 就会立即拿到锁
t2 打印 "t2 notify 之前",执行 notify,唤醒 t1(此时 t1 就从 WAITING 状态恢复回来)
但是由于 t2 还未释放锁,t1 WAITING 状态恢复后,会尝试获取锁,此时会处于阻塞 BLOCKED 状态(由于锁竞争引起的)
t2 执行完 "t2 notify 之后",就会释放锁,且 t2 执行完毕
此时 t1 的wait 就能够获取到锁,并继续执行,打印 "t1 结束等待"
由于,我们也可以知道:当前线程在执行 notify() 方法后,并不会立刻释放锁,而是等 synchronized 代码块执行完后,才会释放锁
如果有多个线程等待,则由线程调度器随机挑选一个处于 wait 状态的线程
notifyAll() 方法
notify() 方法只能唤醒其中一个等待的线程,而使用 notifyAll() 方法可以一次唤醒所有等待线程
public class ThreadDemo18 {public static void main(String[] args) {Object locker = new Object();Thread t1 = new Thread(() -> {System.out.println("t1 进入等待之前");synchronized (locker) {// 进入等待try {locker.wait();} catch (InterruptedException e) {e.printStackTrace();}}System.out.println("t1 结束等待");});Thread t2 = new Thread(() -> {try {Thread.sleep(1000);synchronized (locker) {System.out.println("t2 notify 之前");locker.notifyAll();System.out.println("t2 notify 之后");}} catch (InterruptedException e) {e.printStackTrace();}});Thread t3 = new Thread(() -> {System.out.println("t3 进入等待之前");synchronized (locker) {// 进入等待try {locker.wait();} catch (InterruptedException e) {e.printStackTrace();}}System.out.println("t3 结束等待");});t1.start();t2.start();t3.start();}
}

我们可以看到,notifyAll() 同时唤醒了 t1 和 t3 线程,但是,虽然同时唤醒了 2 个线程,但是这 2 个线程需要竞争锁,因此,并不是同时执行的,而是有先后顺序的执行
nofity 和 notifyAll
notify() 只会唤醒等待队列中的一个线程(由线程调度器随机挑选一个处于 wait 状态的线程),其他线程仍处于 wait 状态
notifyAll() 则会将等待队列中的线程全都唤醒,此时,这些线程需要重新竞争锁,谁先拿到锁,谁后拿到锁,也是不确定的
wait 和 notify
wait 和 notify/notifyAll 彼此之间是通过 object 对象联系起来的
若:
locker1.wait()
locker2.notify()
此时,是无法唤醒使用 locker1.wait() 的线程的,必须两个对象一致才能唤醒,调用 notify 使用的是哪个对象,就会唤醒哪个对象
wait 和 sleep 的区别
wait 和 sleep 都能够让线程放弃执行一段时间,但,wait 是用于线程之间的通信,而 sleep 则是让线程阻塞一段时间
wait 和 sleep 都可以被提前唤醒,wait 通过 notify 唤醒,sleep 通过 interrupt 唤醒,但是
使用 wait 时,一般都是在不确定要等多少时间的前提下使用的(超时时间是用来 "兜底" 的,防止出现 死等)
而使用 sleep 是需要知道需要等多少时间的前提下使用的,虽然能够提前唤醒,但通过异常进行唤醒,此时,大概率说明程序出现了一些特殊情况
此外,
wait() 需要搭配 synchronized 使用,但 sleep() 不需要
wait() 是 Object 提供的方法,sleep() 是 Thread 提供的静态方法
wait 和 join 的区别
同样的,wait 和 join 都能让线程放弃执行一段时间,等待其他线程先执行,但是,wait 是等到 notify 唤醒后,解除 wait 状态,然后参与到锁竞争中;而 join 需要等到其他线程执行完,才会继续执行
当一个线程调用 wait 方法时,会同步释放锁,然后该线程进入等待 状态,其他线程会竞争这把锁,得到锁的线程继续执行
而一个线程运行过程中调用 另一个线程的 join 方法时,当前线程就会停止执行,一直等到另一个线程执行完毕,才会继续执行
wait() 需要搭配 synchronized 使用,但 join() 不需要
wait() 是 Object 提供的方法,join() 是 Thread 提供的方法
相关文章:
wait 和 notify
目录 wait() 方法 notify() 方法 notifyAll() 方法 nofity 和 notifyAll wait 和 notify wait 和 sleep 的区别 wait 和 join 的区别 由于线程之间是抢占式执行的,因此,线程之间执行的先后顺序难以预知,但是,在实际开发中&…...
docker 启动 mongo,redis,nacos.
docker run --name mymongodb -e MONGO_INITDB_ROOT_USERNAMEadmin -e MONGO_INITDB_ROOT_PASSWORDXiaoyusadsad -p 27017:27017 -v /path/to/mongo-data:/data/db -d mongodb/mongodb-community-server:4.4.18-ubuntu2004-v 的目录必须是绝对目录 目录必须 chmod 777 /path/…...
Docker Swarm 搭建
Docker Swarm 搭建 1. 环境介绍 操作系统Centos 7Centos 7Centos 7内核版本Linux 3.10.0-957.el7.x86_64Linux 3.10.0-957.el7.x86_64Linux 3.10.0-957.el7.x86_64主机名称swarm-managerswarm-worker1swarm-worker2IP192.168.1.100192.168.1.200192.168.1.250Docker Domain20…...
浅述TSINGSEE青犀EasyCVR视频汇聚平台与海康安防平台的区别对比
在我们的很多项目中都遇到过用户的咨询:TSINGSEE青犀EasyCVR视频汇聚平台与海康平台的区别在哪里?确实,在安防视频监控领域,EasyCVR视频汇聚平台与海康威视平台是两个备受关注的选择。它们各自具有独特的功能和优势,适…...
设计模式系列:策略模式的设计与实践
一、背景 策略模式(Strategy Pattern)是一种行为设计模式,它定义了一系列的算法,并将每一个算法封装起来,使它们可以相互替换。策略模式让算法的变化独立于使用算法的客户。 二、结构 策略模式主要包含三个角色&…...
数据挖掘之数据预处理
数据预处理是数据挖掘中的一个关键步骤,它的主要目的是对原始数据进行清洗、转换和格式化,以确保其质量和一致性,从而为后续的数据挖掘任务(如分类、回归、聚类等)提供可靠的数据基础。数据预处理一般包括以下几个主要…...
RocketMQ核心知识点整理,值得收藏!
1. 基本概念 Topic: 消息类别的集合,如订单消息发送到order_topic。标签(Tag): 同一Topic下区分不同消息的标志,实现精细化消息管理。ConsumeGroup: 消息消费组,可订阅多个Topic,一个Topic可被多个消费组订…...
微信小程序骨架屏
骨架屏是常用的一种优化方案,针对于页面还未加载完时给用户的一种反馈方式。如果自己要写骨架屏有点复杂因为页面的元素过多且不稳定,这边直接使用微信开发工具生成骨架屏。也不只有微信开发工具有像常用的抖音开发工具,字节开发工具都有对应…...
Window下node安装以及配置
在 Windows 下安装 Node.js 非常简单,你可以通过官方提供的安装程序或者使用多版本管理工具(如 NVM-Win)来进行安装。下面是两种方法的具体步骤: 1. 安装 Node.js程序 步骤如下: 访问官方网站: 访问 Node…...
校园疫情防控系统--论文pf
TOC springboot432校园疫情防控系统--论文pf 课题的来源 2019年在我国武汉爆发了一场规模非常庞大、传播速度十分迅速、对人体危害及其严重的新冠肺炎疫情。引发此次急性感染性新冠肺炎疫情的冠状病毒传播性较强,其传播主要是通过呼吸道飞沫和密切接触这两个途径…...
在Debian 9上使用Apt安装Java的方法
前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。 介绍 Java 和 JVM(Java 虚拟机)是许多软件的必备组件,包括 Tomcat、Jetty、Glassfish、Cassandra 和…...
人工智能在网络安全中的三大支柱
人工智能 (AI) 席卷了网络安全行业,各种供应商都在努力将 AI 融入其解决方案中。但 AI 与安全之间的关系不仅仅在于实现 AI 功能,还在于攻击者和防御者如何利用该技术改变现代威胁形势。它还涉及如何开发、更新和保护这些 AI 模型。如今,网络…...
rk3568mpp终端学习笔记
RK3568Terminal封装MppGraph 通过脚本取和设置音量/zigsun/bin/linux/bin.debug.Linux.rk3568/get_record_voice_value.sh /zigsun/bin/linux/bin.debug.Linux.rk3568/set_record_voice_value.sh class RK3568Terminal : public IAVLinkManager, p…...
【C++继承】赋值兼容转换作用域派生类的默认成员函数
1.继承的概念 继承(inheritance)机制是面向对象程序设计使代码可以复用的最重要的手段,它允许程序员在保持原有类特性的基础上进行扩展,增加功能,这样产生新的类,称派生类(或子类)。继承呈现了面向对象程序设计的层次结构…...
HTML5+JavaScript绘制彩虹和云朵
HTML5JavaScript绘制彩虹和云朵 彩虹,简称虹,是气象中的一种光学现象,当太阳光照射到半空中的水滴,光线被折射及反射,在天空上形成拱形的七彩光谱,由外圈至内圈呈红、橙、黄、绿、蓝、靛、紫七种颜色。事实…...
MySQL——单表查询(二)按条件查询(2)带 IN 关键字的查询
IN 关键字用于判断某个字段的值是否在指定集合中,如果字段的值在集合中,则满足条件,该字段所在的记录将被查询出来。其语法格式如下所示: SELECT *|字段名 1,字段名 2,… FROM 表名 WHERE 字段名 [NOT〕IN(元素 1,元素 2,…) 在上…...
【mysql】mysql 用户管理---创建、权限管理等等
本站以分享各种运维经验和运维所需要的技能为主 《python零基础入门》:python零基础入门学习 《python运维脚本》: python运维脚本实践 《shell》:shell学习 《terraform》持续更新中:terraform_Aws学习零基础入门到最佳实战 《k8…...
本地服务器物理机中redis设置、取消密码
1.服务器物理机上redis的操作【服务器中操作】 (1)首先先看一下当前运行中的redis实例: [rootiZuf67k70ucx14s6zcv54dZ var]# ps aux | grep redis-server因为我这里有两个实例在运行,即物理机上的redis和docker中的redis&…...
关于xilinx的FFTIP的使用和仿真
工具:vivado2018.3,modelsim10.6d 场景:在进行数据进行频谱分析的时候,使用FPGA来完成FFT的计算可以加快数据的计算速度。 下面使用仿真完成DDS产生的数据的FFT以及IFFT。原始数据使用DDSIP产生,通过IP产生的波形数据…...
ant design pro 如何去保存颜色
上图 就是实现这样的效果 后端是这样的,这个颜色肯定是存到字符串里的 这是第一步 import mongoose, { Schema, Document } from mongoose;interface IDiscountCard extends Document {title: string;subtitle: string;image: string;shopUrl: string;bgColor: s…...
【Oracle APEX开发小技巧12】
有如下需求: 有一个问题反馈页面,要实现在apex页面展示能直观看到反馈时间超过7天未处理的数据,方便管理员及时处理反馈。 我的方法:直接将逻辑写在SQL中,这样可以直接在页面展示 完整代码: SELECTSF.FE…...
JVM垃圾回收机制全解析
Java虚拟机(JVM)中的垃圾收集器(Garbage Collector,简称GC)是用于自动管理内存的机制。它负责识别和清除不再被程序使用的对象,从而释放内存空间,避免内存泄漏和内存溢出等问题。垃圾收集器在Ja…...
学习STC51单片机31(芯片为STC89C52RCRC)OLED显示屏1
每日一言 生活的美好,总是藏在那些你咬牙坚持的日子里。 硬件:OLED 以后要用到OLED的时候找到这个文件 OLED的设备地址 SSD1306"SSD" 是品牌缩写,"1306" 是产品编号。 驱动 OLED 屏幕的 IIC 总线数据传输格式 示意图 …...
让AI看见世界:MCP协议与服务器的工作原理
让AI看见世界:MCP协议与服务器的工作原理 MCP(Model Context Protocol)是一种创新的通信协议,旨在让大型语言模型能够安全、高效地与外部资源进行交互。在AI技术快速发展的今天,MCP正成为连接AI与现实世界的重要桥梁。…...
Rapidio门铃消息FIFO溢出机制
关于RapidIO门铃消息FIFO的溢出机制及其与中断抖动的关系,以下是深入解析: 门铃FIFO溢出的本质 在RapidIO系统中,门铃消息FIFO是硬件控制器内部的缓冲区,用于临时存储接收到的门铃消息(Doorbell Message)。…...
稳定币的深度剖析与展望
一、引言 在当今数字化浪潮席卷全球的时代,加密货币作为一种新兴的金融现象,正以前所未有的速度改变着我们对传统货币和金融体系的认知。然而,加密货币市场的高度波动性却成为了其广泛应用和普及的一大障碍。在这样的背景下,稳定…...
Linux 中如何提取压缩文件 ?
Linux 是一种流行的开源操作系统,它提供了许多工具来管理、压缩和解压缩文件。压缩文件有助于节省存储空间,使数据传输更快。本指南将向您展示如何在 Linux 中提取不同类型的压缩文件。 1. Unpacking ZIP Files ZIP 文件是非常常见的,要在 …...
HybridVLA——让单一LLM同时具备扩散和自回归动作预测能力:训练时既扩散也回归,但推理时则扩散
前言 如上一篇文章《dexcap升级版之DexWild》中的前言部分所说,在叠衣服的过程中,我会带着团队对比各种模型、方法、策略,毕竟针对各个场景始终寻找更优的解决方案,是我个人和我司「七月在线」的职责之一 且个人认为,…...
数据结构第5章:树和二叉树完全指南(自整理详细图文笔记)
名人说:莫道桑榆晚,为霞尚满天。——刘禹锡(刘梦得,诗豪) 原创笔记:Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊) 上一篇:《数据结构第4章 数组和广义表》…...
flow_controllers
关键点: 流控制器类型: 同步(Sync):发布操作会阻塞,直到数据被确认发送。异步(Async):发布操作非阻塞,数据发送由后台线程处理。纯同步(PureSync…...
