这篇文章主要介绍了简单了解java等待唤醒机制原理及使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
这是一篇走心的填坑笔记,自学Java的几年总是在不断学习新的技术,一路走来发现自己踩坑无数,而填上的坑却屈指可数。突然发现,有时候真的不是几年工作经验的问题,有些东西即使工作十年,没有用心去学习过也不过是一个10年大坑罢了(真实感受)。
刚开始接触多线程时,就知道有等待/唤醒这个东西,写过一个demo就再也没有看过了,至于它到底是个什么东西,或者说它能解决什么样的问题,估计大多数人和我一样都是模棱两可。这次笔者就尝试带你搞懂等待/唤醒机制,读完本文你将get到以下几点:
一,循环等待问题
假设今天要发工资,强老板要去吃一顿好的,整个就餐流程可以分为以下几个步骤:
public static void main(String[] args) { // 是否还有包子 AtomicBoolean hasBun = new AtomicBoolean(); // 包子铺老板 new Thread(() -> { try { // 一直循环查看是否还有包子 while (true) { if (hasBun.get()) { System.out.println("老板:检查一下是否还剩下包子..."); Thread.sleep(3000); } else { System.out.println("老板:没有包子了, 马上开始制作..."); Thread.sleep(1000); System.out.println("老板:包子出锅咯...."); hasBun.set(true); } } } catch (InterruptedException e) { e.printStackTrace(); } }).start(); new Thread(() -> { System.out.println("小强:我要买包子..."); try { // 每隔一段时间询问是否完成 while (!hasBun.get()) { System.out.println("小强:包子咋还没做好呢~"); Thread.sleep(3000); } System.out.println("小强:终于吃上包子了...."); } catch (InterruptedException e) { e.printStackTrace(); } }).start(); }
在上文代码中存在一个很大的问题,就是老板需要不断的去检查是否还有包子,而客户则需要隔一段时间去看催一下老板,这显然时不合理的,这就是典型的循环等待问题。
这种问题的代码中通常是如下这种模式:
while (条件不满足) { Thread.sleep(3000); } doSomething();
对应到计算机中,则暴露了一个问题:不断通过轮询机制来检测条件是否成立, 如果轮询时间过小则会浪费CPU资源,如果间隔过大,又导致不能及时获取想要的资源。
二,等待/唤醒机制
为了解决循环等待消耗CPU以及信息及时性问题,Java中提供了等待唤醒机制。通俗来讲就是由主动变为被动, 当条件成立时,主动通知对应的线程,而不是让线程本身来询问。
2.1 基本概念
等待/唤醒机制,又叫等待通知(笔者更喜欢叫唤醒而非通知),是指线程A调用了对象O的wait()方法进入了等待状态,而另一个线程调用了O的notify()或者notifyAll()方法,线程A收到通知后从对象O的wait()方法返回,进而执行后续操作。
上诉过程是通过对象O,使得线程A和线程B之间进行通信, 在线程中调用了对象O的wait()方法后线程久进入了阻塞状态,而在其他线程中对象O调用notify()或notifyAll方法时,则会唤醒对应的阻塞线程。
2.2 基本API
等待/唤醒机制的相关方法时任意Java对象具备的,因为这些方法被定义在所有Java对象的超类Object中。
notify: 通知一个在对象上等待的线程,使其从wait()方法返回,而返回的前提时该线程获取到对象的锁
notifyAll: 通知所有等待在该对象上的线程
wait: 调用此方法的线程进入阻塞等待状态,只有等待另外线程的通知或者被中断才会返回,调用wait方法会释放对象的锁
wait(long) : 等待超过一段时间没有被唤醒就超时自动返回,单位时毫秒。
2.3 用等待唤醒机制优化循环等待
public static void main(String[] args) { // 是否还有包子 AtomicBoolean hasBun = new AtomicBoolean(); // 锁对象 Object lockObject = new Object(); // 包子铺老板 new Thread(() -> { try { while (true) { synchronized (lockObject) { if (hasBun.get()) { System.out.println("老板:包子够卖了,打一把王者荣耀"); lockObject.wait(); } else { System.out.println("老板:没有包子了, 马上开始制作..."); Thread.sleep(3000); System.out.println("老板:包子出锅咯...."); hasBun.set(true); // 通知等待的食客 lockObject.notifyAll(); } } } } catch (InterruptedException e) { e.printStackTrace(); } }).start(); new Thread(() -> { System.out.println("小强:我要买包子..."); try { synchronized (lockObject) { if (!hasBun.get()) { System.out.println("小强:看一下有没有做好, 看公众号cruder有没有新文章"); lockObject.wait(); } else { System.out.println("小强:包子终于做好了,我要吃光它们...."); hasBun.set(false); lockObject.notifyAll(); System.out.println("小强:一口气把店里包子吃光了, 快快乐乐去板砖了~~"); } } } catch (InterruptedException e) { e.printStackTrace(); } }).start(); }
上述流程,减少了轮询检查的操作,并且线程调用wait()方法后,会释放锁,不会消耗CPU资源,进而提高了程序的性能。
三,等待唤醒机制的基本范式
等待、唤醒是线程间通信的手段之一,用来协调多个线程操作同一个数据源。实际应用中通常用来优化循环等待的问题,针对等待方和通知方,可以提炼出如下的经典范式。
需要注意的是,在等待方执行的逻辑中,一定要用while循环来判断等待条件,因为执行notify/notifyAll方法时只是让等待线程从wait方法返回,而非重新进入临界区
/** * 等待方执行的逻辑 * 1. 获取对象的锁 * 2. 检查条件,如果条件不满足,调用对象的wait方法,被通知后重新检查条件 * 3. 条件满足则执行对应的逻辑 */ synchronized(对象){ while(条件不满足){ 对象.wait() } doSomething(); } /** * !! 通知方执行的逻辑 * 1. 获取对象的锁 * 2. 改变条件 * 3. 通知(所有)等待在对象上的线程 */ synchronized(对象){ 条件改变 对象.notify(); }
这个编程范式通常是针对典型的通知方和等待方,有时双方可能具有双重身份,即使等待方又是通知方,正如我们上文中的案例一样。
四,notify/notifyAll不释放锁
相信这个问题有半数工程师都不知道,当执行wait()方法,锁自动被释放;但执行完notify()方法后,锁不会释放,而是要执行notify()方法所在的synchronized代码块后才会释放。这一点很重要,也是很多工程师容易忽略的地方。
lockObject.notifyAll(); System.out.println("小强:一口气把店里包子吃光了, 快快乐乐去板砖了~~");
案例代码中,故意设置成先notifyAll,然后在打印;上文图中的结果也印证了了我们的描述,感兴趣的小伙伴可以动手执行一下案例代码哦。
五,等待、唤醒必须先获取锁
在等待、唤醒编程范式中的wait,notify,notifyAll方法往往不能直接调用, 需要在获取锁之后的临界区执行
并且只能唤醒等待在同一把锁上的线程。
当线程调用wait方法时会被加入到一个等待队列,当执行notify时会唤醒队列中第一个等待线程(等待时间最长的线程),而调用notifyAll时则会唤醒等待线程中所有的等待线程。
六,sleep不释放锁 而wait 释放#
在用等待唤醒机制优化循环等待的过程中,有一个重要的特征就是原本的sleep()方法用wait()方法取代,他们的最大的区别在于wait方法会释放锁,而sleep不会,除此之外,还有个重要的区别,sleep是Thread的方法,可以在任意地方执行;而wait是Object对象的方法,必须在synchronized代码块中执行。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持小牛知识库。
本文向大家介绍Java等待唤醒机制原理实例解析,包括了Java等待唤醒机制原理实例解析的使用技巧和注意事项,需要的朋友参考一下 这篇文章主要介绍了Java等待唤醒机制原理实例解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 线程的状态 首先了解一下什么是线程的状态,线程状态就是当线程被创建(new),并且启动(start)后,它不是一启动就进入
我想做一个小练习来习惯等待/通知。我想做的是简单地启动一个线程,然后用等待让它进入睡眠状态,用通知唤醒它,多次。 我的代码是: 我希望这会是这样 相反,这样做: 所以。。。通知似乎没有唤醒打印机线程? 这不应该是一个死锁,因为通过等待,我释放了所有的锁,所以主服务器不应该有任何对打印机的锁,打印机应该能够唤醒并打印。 我做错了什么?
本文向大家介绍简单了解python的内存管理机制,包括了简单了解python的内存管理机制的使用技巧和注意事项,需要的朋友参考一下 Python引入了一个机制:引用计数。 引用计数 python内部使用引用计数,来保持追踪内存中的对象,Python内部记录了对象有多少个引用,即引用计数,当对象被创建时就创建了一个引用计数,当对象不再需要时,这个对象的引用计数为0时,它被垃圾回收。 总结一下对象会在
本文向大家介绍Java多线程基础 线程的等待与唤醒(wait、notify、notifyAll),包括了Java多线程基础 线程的等待与唤醒(wait、notify、notifyAll)的使用技巧和注意事项,需要的朋友参考一下 本篇我们来研究一下 wait() notify() notifyAll() 。 DEMO1: wait() 与 notify() DEMO1 输出: 注意: 使用 wait
本文向大家介绍简单了解mybatis拦截器实现原理及实例,包括了简单了解mybatis拦截器实现原理及实例的使用技巧和注意事项,需要的朋友参考一下 这篇文章主要介绍了简单了解mybatis拦截器实现原理及实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 例行惯例,先看些基本概念: 1 拦截器的作用就是我们可以拦截某些方法的调用,在目标方法前后加
本文向大家介绍Java泛型机制必要性及原理解析,包括了Java泛型机制必要性及原理解析的使用技巧和注意事项,需要的朋友参考一下 泛型程序设计(Generic Programming) “使用泛型机制编写的程序代码要比那些杂乱地使用Object变量,然后再进行强制类型转换的代码具有更好的安全性和可读性。泛型对于集合类尤其有用。” 1.意义、必要性、重要性 泛型程序设计 意味着编写的代码可以被很多不同
本文向大家介绍Java SPI机制原理及代码实例,包括了Java SPI机制原理及代码实例的使用技巧和注意事项,需要的朋友参考一下 SPI的全名为:Service Provider Interface,大多数开发人员可能不熟悉,因为这个是针对厂商或者插件的。在java.util.ServiceLoader的文档里有比较详细的介绍。 简单的总结下 Java SPI 机制的思想。我们系统里抽象的各个模
问题内容: 哈罗我已经整天调试了我的代码,但是我看不出哪里出了问题。 我在主线程上使用SerialPortEventListener,在工作线程中,我有一个客户端套接字与服务器通信。由于到达此工作线程之后,我仍然需要在主线程中完成一些总结工作,因此我想创建一个“伪线程”,在主线程中等待,直到从侦听器onEvent方法通知它为止。 但是这个伪线程似乎一直在等待。 我检查了锁定的线程,它们在Runna