多线程编程7——wait和notify、notifyAll

线程最大的问题就是抢占式执行,随机调度。可以通过一些API让线程主动阻塞,主动放弃CPU,从而控制线程之间的执行顺序。比如:join,sleep,wait和notify、notifyAll

前面章节已经介绍过 join 和 sleep了,那么接下来我们来介绍介绍wait 和 notify、notifyAll

wait 和 notify、notifyAll 是 Object 类中的方法。Java 里任意一个对象都有这两个方法。

1、wait()

正确使用 wait 的代码如下(搭配 synchronized 使用):

public class ThreadDemo15 {public static void main(String[] args) {Object object = new Object();//这个线程负责等待Thread t1 = new Thread(()->{System.out.println("wait 之前");try {synchronized (object){object.wait();}} catch (InterruptedException e) {throw new RuntimeException(e);}System.out.println("wait 之后");});t1.start();}
}

在线程中调用wait,线程进入阻塞等待状态,比如在 线程t1 中调用 wait,就是让线程t1 进入阻塞等待状态(WAITING)。有有参和无参两个版本。无参版本,就是死等,直到被通知唤醒。有参版本,就是规定了一个最大的等待时间,超过时间就不等了。

wait 操作有3步:1、先释放2、进行阻塞等待 3、收到通知(notify)之后,重新尝试获取锁,并在获取锁后,继续往下执行

所以,wait 要搭配 synchronizwd 来使用,要先获取到锁,才能执行 wait 释放锁,否则就会报异常(IllegalMonitorStateException,非法的锁状态异常)。就像,你还没有女朋友,是个单身狗呢,就想分手,你要和谁分啊。

无论在线程中是通过哪个锁对象 wait 的,这个线程都会处于阻塞等待状态,且阻塞在 synchronized 代码块内,此时是释放了锁的,其他线程是可以对 锁对象 加锁的。

wait 结束等待的条件如下:

1、如果 wait 有参数,超过时间就不等了

2、interrupt 触发异常唤醒 wait

3、其他线程调用了该对象的notify方法(这个对象就是锁对象),通知在 同一个对象上 等待的线程,此时在同一个对象上等待的线程就会重新尝试加锁。(只能有一个线程会加锁成功)

报 非法的锁状态异常 的代码如下:

public class ThreadDemo15 {public static void main(String[] args) {Object object = new Object();//这个线程负责等待Thread t1 = new Thread(()->{System.out.println("wait 之前");try {object.wait();} catch (InterruptedException e) {throw new RuntimeException(e);}System.out.println("wait 之后");});t1.start();}
}

没有加锁,就想执行 wait 释放锁,报异常。

2、notify()

通知在同一个对象上等待的线程也就是说,此处 notify 的对象要和 wait 的对象要相同,如果通知和等待是两个不同对象,通知就不会生效,wait就不会唤醒。

也要结合 synchronized关键字,要先获取到锁,才能通知。

public class ThreadDemo15 {public static void main(String[] args) throws InterruptedException {Object object = new Object();//这个线程负责等待Thread t1 = new Thread(()->{System.out.println("wait 之前");try {synchronized (object){object.wait();}} catch (InterruptedException e) {throw new RuntimeException(e);}System.out.println("wait 之后");});//这个线程负责通知Thread t2 = new Thread(()->{System.out.println("notify之前");synchronized (object){object.notify();}System.out.println("notify之后");});t1.start();Thread.sleep(500);t2.start();}
}

四个地方的对象都要相同,比如上面,都是object。wait使用的对象要和notify使用的对象相同,不然就通知了个寂寞,notify不会有任何效果。notify 只能唤醒在同一个对象上等待的线程。

3、总结

1、wait和notify都要先获取到锁才能使用。获取到锁后,使用wait,线程会先释放锁对象,然后阻塞等待;使用notify,会通知在同一个锁对象上等待的线程,如果当前有多个线程在等待同一个锁对象,会随机唤醒一个等待的线程;而notifyAll,是所有线程都唤醒,这些线程再一起竞争锁。

2、wait和sleep的区别:

(1)虽然wait和sleep都能被提前唤醒,wait是使用notify唤醒,sleep是使用interrupt唤醒。但是notify唤醒wait,不会有任何异常,是正常的业务逻辑。而interrupt唤醒sleep,则是出现了异常,表示一个出问题的逻辑。

(2)wait需要搭配synchronized使用,sleep不需要

(3)wait是Object的方法,sleep是Thread的方法

3、wait和join的区别:

wait可以使用notify提前唤醒,但是join则必须得等这个线程彻底执行完,下个线程才能执行,不能提前唤醒。

4、wait和notify的使用 

wait和notify可以对多个线程的执行顺序进行控制。wait会让调用的线程进行阻塞,通过其他线程的notify进行通知。

如:三个线程,分别只能打印A,B,C,请你写个代码,保证这三个线程,固定按照ABC这样的顺序来打印

 public static void main(String[] args) throws InterruptedException {//让 C 后于 B 打印,让 B后于 A打印Object object1 = new Object();Object object2 = new Object();Thread t1 = new Thread(()->{System.out.println("A");synchronized (object2) {object2.notify();}});Thread t2 = new Thread(()->{synchronized (object2){try {object2.wait();} catch (InterruptedException e) {throw new RuntimeException(e);}}System.out.println("B");synchronized (object1){object1.notify();}});Thread t3 = new Thread(()->{synchronized (object1){try {object1.wait();} catch (InterruptedException e) {throw new RuntimeException(e);}}System.out.println("C");});t2.start();t3.start();//防止notify通知了个寂寞,wait没人唤醒了Thread.sleep(1000);t1.start();}

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

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

相关文章

计算机网络-IPv6地址规范与分类

昨天学习了IPv6的基础概念,了解了IPv6的由来以及地址格式,今天继续学习下IPv6的地址分类与表示。 一、IPv6地址缩写规范 IPv6地址的长度为128 bit。一般用冒号分割为8段,每一段16 bit,每一段内用十六进制表示。 IPv6地址格式 那12…

【kettle004】kettle访问本地MySQL数据库并处理数据至execl文件

一直以来想写下基于kettle的系列文章,作为较火的数据ETL工具,也是日常项目开发中常用的一款工具,最近刚好挤时间梳理、总结下这块儿的知识体系。 熟悉、梳理、总结下MySQL关系数据库相关知识体系 kettle访问MySQL数据库并处理数据至execl文件…

详解SPI、I2C、UART、I2S、GPIO、SDIO、CAN

总线,总线,总要陷进里面。这世界上的信号都一样,但是总线却成千上万,让人头疼。 总的来说,总线有三种:内部总线、系统总线和外部总线。内部总线是微机内部各外围芯片与处理器之间的总线,用于芯片一级的互连;而系统总线是微机中各插件板与系统板之间的总线,用于插件板一…

为什么要进行人脸识别?

人脸识别技术被广泛应用于各种场景和行业,其主要目的包括但不限于以下几点: 1. **安全验证**:人脸识别可以用作身份验证的一种方式,确保只有授权人员才能进入特定的区域或访问敏感信息。例如,用于解锁手机或电脑、进入…

Linux---为什么会有粘滞位?

在前面已经讲过目录的rwx权限: 可读权限(r): 如果目录没有可读权限, 则无法用ls等命令查看目录中的文件内容. 有可写权限(w):如果目录没有可写权限,则无法在目录中创建文件, 也无法在目录中删除文件.可执行权限(x): 如果目录没有可执行权限, 则无法cd到…

D-Wave 推出快速退火功能,扩大量子计算性能增益

内容来源:量子前哨(ID:Qforepost) 文丨浪味仙 排版丨沛贤 深度好文:1400字丨6分钟阅读 摘要:量子计算公司 D-Wave 宣布在其 Leap™ 实时量子云服务中的所有量子处理单元 (QPU) 上推出新的快速退火功能。…

Java 网络编程之TCP(五):分析服务端注册OP_WRITE写数据的各种场景(二)

接上文 二、注册OP_WRITE写数据 服务端代码: import java.io.IOException; import java.net.InetSocketAddress; import java.nio.ByteBuffer; import java.nio.channels.SelectableChannel; import java.nio.channels.SelectionKey; import java.nio.channels.S…

RFC 6071: IP Security (IPsec) and Internet Key Exchange (IKE) Document Roadmap

![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/96882d1fb67b4383bc77c4dd421f7b

【一刷《剑指Offer》】面试题 10:二进制中 1 的个数

力扣对应题目链接:191. 位1的个数 - 力扣(LeetCode) 牛客对应题目链接:二进制中1的个数_牛客题霸_牛客网 一、《剑指Offer》内容 核心考点 :二进制计算。 二、分析问题 1、循环检查二进制位 可以直接循环检查给定数字…

Springboot实现国际化以及部署Linux不生效问题

1、在application.properties 添加以下配置: #国际化配置 spring.messages.basenamei18n/messages/messages spring.messages.fallback-to-system-localefalse 2、添加配置文件在 resources目录下 如下图所示: 这个国际化文件命名有个坑,必须…

胸部X光图像分类:因果视角

Chest X-ray Image Classification: A Causal Perspective 摘要 这篇论文提出了一种新的方法来处理胸部X射线(CXR)分类问题,将因果推理融入到基于深度学习的方法中。尽管近年来深度学习在CXR分类方面取得了许多进展,但确保这些算…

【接口测试】JMeter接口关联测试

‍‍1 前言 我们来学习接口管理测试,这就要使用到JMeter提供的JSON提取器和正则表达式提取器了,下面我们来看看是如何使用的吧。 2 JSON提取器 1、添加JSON提取器 在线程组右键 > 添加 > 后置处理器 > JSON提取器 2、JSON提取器参数说明 N…

在浏览器中查看Revit模型:原理及实现

Greg Schleusner AIA 有一天向我提到,业内许多人认为带有 SVF2 的 Autodesk Viewer 是建筑模型最强大、性能最强的在线查看器之一,并且想知道开源 VIM 查看器和格式如何比较。 可以使用NSDT 3DConvert的 Revit插件 上传你的模型并在线查看预览&#xff…

怎么通过PHP语言实现远程控制棋牌室

怎么通过PHP语言实现远程控制棋牌室呢? 本文描述了使用PHP语言调用HTTP接口,实现控制棋牌室,通过专用的包间控制器,来实现包间内所有电器以及门锁的独立控制。 可选用产品:可根据实际场景需求,选择对应的规…

BetterDisplay Pro for Mac:显示器校准软件

BetterDisplay Pro for Mac是一款出色的显示器校准软件,旨在提升你的视觉体验。它提供了准确的显示器参数调整,包括亮度、对比度、色温和色域等,让你的显示器呈现更真实、清晰、细腻的图像。此外,软件还提供多种预设模式和自定义选…

遥感雷达波段的原理及应用

雷达波段是不同波长的组。每一种都有其独特的穿透地球表面的能力。它们还可以揭示环境的不同方面。 雷达频段在电磁频谱内具有特定的频率范围。这些波段由 L-、S-、C- 和 X-波段等字母表示。稍后会详细介绍这一点。 什么是合成孔径雷达? 合成孔径雷达 (SAR) 是一…

后台图书管理系统:SSM整合开发案例

代码已经完善,文章内容会在后续补充 代码地址(https://javazhang.lanzn.com/ig8tf1wd2aba 密码:1234) 1.1 .SpringConfig 代码 Configuration ComponentScan({"com.itheima.service" }) PropertySource("classpath:jdbc.properties") Import({JdbcConfig…

【STM32+HAL】读取电池电量

一、准备工作 有关CUBEMX的初始化配置,参见我的另一篇blog:【STM32HAL】CUBEMX初始化配置 有关定时器触发ADC模式配置,详见【STM32HAL】ADC采集波形实现 有关软件触发ADC模式配置,详见【STM32HAL】三轴按键PS2摇杆 二、所用工具…

如何替代传统的方式,提高能源企业敏感文件传输的安全性?

能源行业是一个关键的基础设施领域,它涉及能源的勘探、开采、生产、转换、分配和消费。随着全球经济的发展和人口的增长,能源需求持续上升,这对能源行业的可持续发展提出了挑战。能源行业的传输场景多种多样,需要重点关注能源企业…

【热门话题】Chrome 插件研发详解:从入门到实践

🌈个人主页: 鑫宝Code 🔥热门专栏: 闲话杂谈| 炫酷HTML | JavaScript基础 ​💫个人格言: "如无必要,勿增实体" 文章目录 Chrome 插件研发详解:从入门到实践一、引言二、Chrome 插件基础概念…