通俗的解释JAVA wait/notify机制
生活中,我们常遇到需要等待的场景,例如去银行办事,在没轮到自己之前需要一直等待,但是如果需要自己每隔几秒钟就去柜台前看看状况,那肯定是种非常低效和令人恼火的体验。而实际的情况是,接待员会让您拿个号,说"请稍等一会"(wait); 当排到时,语言和大屏幕会提示"请XXX号到N号柜台办理"(notify)。
wait/notify机制也正是处理这样的场景:线程继续执行需要等待某个条件的变化,这个条件由另一个任务改变,如果一直空循环检查条件变化,是一种不良的CPU使用方式,这时候可以调用wait()将任务挂起,在其他线程调用了notify()或notifyAll()时,任务被唤醒并检查条件的变化。
这个过程中,锁的持有发生了变化。介绍wait/notify最常用的例子是生产者和消费者,设想你去饭馆吃饭,叫来服务员说,把我的宫保鸡丁端上来吧。这时候你获得了服务员的锁,在解决你的事情前,服务员不能去做别的事。(同一时间,厨师可能已经做好了宫保鸡丁,等服务员来端,但是服务员在和你说话,厨师束手无策(等待锁)。)服务员没有宫保鸡丁,只能对你说:您稍等一下,我去厨房催催。服务员调用了wait()方法,你只好释放锁,服务员回到厨房,厨师怒气冲冲的喊(获得锁),宫保鸡丁好了,端走。
下面用程序演示这一场景:
- public class Waiter {
- private String dishes = null;
- public synchronized String getDishes() {
- System.out.printf("顾客获得服务员锁%n");
- while(this.dishes == null) {
- try {
- System.out.printf("顾客取菜,没有菜...顾客线程等待(释放锁)%n");
- wait();
- } catch(InterruptedException ex) {
- ex.printStackTrace();
- }
- }
- String d = this.dishes;
- System.out.printf("顾客取走: %s%n", this.dishes);
- this.dishes = null;
- notifyAll();
- System.out.printf("服务员通知正在等待的线程%n");
- return d;
- }
- public synchronized void setDishes(String dishes) {
- System.out.printf("厨师获得服务员锁%n");
- while(this.dishes != null) {
- try {
- System.out.printf("厨师交菜,服务员已经端了另一份菜...厨师线程等待(释放锁)%n");
- wait();
- } catch(InterruptedException ex) {
- ex.printStackTrace();
- }
- }
- this.dishes = dishes;
- System.out.printf("厨师交菜: %s%n", this.dishes);
- notifyAll();
- System.out.printf("服务员通知正在等待的线程(顾客)%n");
- }
- public static void main(String[] args) throws InterruptedException {
- Waiter busy = new Waiter();
- for(int i = 0; i < 10; i++) {
- Thread consumer = new Thread() {
- public void run() {
- busy.getDishes();
- }
- };
- consumer.start();
- }
- Thread.sleep(100);
- for(int i = 0; i < 10; i++) {
- Thread chef = new Thread() {
- public void run() {
- String dishes = "宫保鸡丁";
- busy.setDishes(dishes);
- }
- };
- chef.start();
- }
- }
- }
运行结果:
- 顾客获得服务员锁
- 顾客取菜,没有菜...顾客线程等待(释放锁)
- 厨师获得服务员锁
- 厨师交菜: 宫保鸡丁
- 服务员通知正在等待的线程(顾客)
- 顾客取走: 宫保鸡丁
- 服务员通知正在等待的线程(厨师)
下面来说明notifyAll的作用。
修改下代码,把厨师和顾客都增加到10个
- public static void main(String[] args) {
- Waiter busy = new Waiter();
- for(int i = 0; i < 10; i++) {
- Thread consumer = new Thread() {
- public void run() {
- busy.getDishes();
- }
- };
- consumer.start();
- }
- for(int i = 0; i < 10; i++) {
- Thread chef = new Thread() {
- public void run() {
- String dishes = "宫保鸡丁";
- busy.setDishes(dishes);
- }
- };
- chef.start();
- }
- }
执行后会发现程序会陷入永久的等待无法结束,这是因为notify()方法只唤醒众多等待的线程中的一个,拿到菜后本应唤醒顾客取走,但是有可能随机唤醒了另一个等待的厨师,没有顾客能取走服务员手中的菜,这时候程序就无法继续下去了。
解决的方法有两种:
1 把notify()改成notifyAll(),唤醒所有等待的线程
2 使用Java.util.concurrent库中的Condition,把等待的线程分为厨师和顾客两个集合,代码如下:
- public class ConditionWaiter {
- private String dishes = null;
- private Lock lock = new ReentrantLock();
- private Condition conConsumer = lock.newCondition();
- private Condition conChef = lock.newCondition();
- public String getDishes() {
- try {
- lock.lock();
- System.out.printf("顾客获得服务员锁%n");
- while(this.dishes == null) {
- try {
- System.out.printf("顾客取菜,没有菜...顾客线程等待%n");
- conConsumer.await();
- } catch(InterruptedException ex) {
- ex.printStackTrace();
- }
- }
- String d = this.dishes;
- System.out.printf("顾客取走:%s%n", this.dishes);
- this.dishes = null;
- conChef.signal();
- System.out.printf("服务员通知正在等待的线程(厨师)%n");
- return d;
- } finally {
- lock.unlock();
- }
- }
- public void setDishes(String dishes) {
- try {
- lock.lock();
- System.out.printf("厨师获得服务员锁%n");
- while(this.dishes != null) {
- try {
- System.out.printf("厨师交菜,服务员已经端了另一份菜...厨师线程等待%n");
- conChef.await();
- } catch(InterruptedException ex) {
- ex.printStackTrace();
- }
- }
- this.dishes = dishes;
- System.out.printf("厨师交菜:%s%n", this.dishes);
- conConsumer.signal();
- System.out.printf("服务员通知正在等待的线程(顾客)%n");
- } finally {
- lock.unlock();
- }
- }
- public static void main(String[] args) throws InterruptedException {
- ConditionWaiter busy = new ConditionWaiter();
- for(int i = 0; i < 10; i++) {
- Thread consumer = new Thread() {
- public void run() {
- busy.getDishes();
- }
- };
- consumer.start();
- }
- Thread.sleep(100);
- for(int i = 0; i < 10; i++) {
- Thread chef = new Thread() {
- public void run() {
- String dishes = "宫保鸡丁";
- busy.setDishes(dishes);
- }
- };
- chef.start();
- }
- }
- }
事实上,wait/notify机制编程模型复杂也运行低效,通常我们应该采取更高级的类库实现类似场景。以下代码是使用BlockingQueue实现线程协作的示例:
- public class BlockingQueueWaiter {
- static BlockingQueue<String> queue = new ArrayBlockingQueue<>(1);
- public static void main(String[] args) throws InterruptedException {
- for(int i = 0; i < 10; i++) {
- Thread consumer = new Thread() {
- public void run() {
- String dishes;
- try {
- System.out.printf("顾客尝试取菜%n");
- dishes = queue.take();
- System.out.printf("顾客取走:%s%n", dishes);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- }
- };
- consumer.start();
- }
- Thread.sleep(100);
- for(int i = 0; i < 10; i++) {
- Thread chef = new Thread() {
- public void run() {
- String dishes = "宫保鸡丁";
- try {
- System.out.printf("厨师尝试交菜%n");
- queue.put(dishes);
- System.out.printf("厨师交菜:%s%n", dishes);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- }
- };
- chef.start();
- }
- }
- }
通俗的解释JAVA wait/notify机制的更多相关文章
- QWaitCondition(和Java的Notify机制非常相像)
QT通过三种形式提供了对线程的支持.它们分别是,一.平台无关的线程类,二.线程安全的事件投递,三.跨线程的信号-槽连接.这使得开发轻巧的多线程Qt程序更为容易,并能充分利用多处理器机器的优势.多线程编 ...
- 为什么JAVA要提供 wait/notify 机制?是为了避免轮询带来的性能损失
wait/notify 机制是为了避免轮询带来的性能损失. 为了说清道理,我们用“图书馆借书”这个经典例子来作解释. 一本书同时只能借给一个人.现在有一本书,图书馆已经把这本书借了张三. 在简单的s ...
- C# 实现java中 wiat/notify机制
最近在学习java,看到wiat/notify机制实现线程通信,由于平时工作用的C#,赶紧用C#方式实现一个demo. Java 代码: import java.util.ArrayList; imp ...
- Java的多线程机制系列:(一)总述及基础概念
前言 这一系列多线程的文章,一方面是个人对Java现有的多线程机制的学习和记录,另一方面是希望能给不熟悉Java多线程机制.或有一定基础但理解还不够深的读者一个比较全面的介绍,旨在使读者对Java的多 ...
- Java 类反射机制分析
Java 类反射机制分析 一.反射的概念及在Java中的类反射 反射主要是指程序可以访问.检测和修改它本身状态或行为的一种能力.在计算机科学领域,反射是一类应用,它们能够自描述和自控制.这类应用通过某 ...
- JAVA 初识类加载机制 第13节
JAVA 初识类加载机制 第13节 从这章开始,我们就进入虚拟机类加载机制的学习了.那么什么是类加载呢?当我们写完一个Java类的时候,并不是直接就可以运行的,它还要编译成.class文件,再由虚拟机 ...
- 沉淀再出发:再谈java的多线程机制
沉淀再出发:再谈java的多线程机制 一.前言 自从我们学习了操作系统之后,对于其中的线程和进程就有了非常深刻的理解,但是,我们可能在C,C++语言之中尝试过这些机制,并且做过相应的实验,但是对于ja ...
- java垃圾回收机制学习总结
最近学习了一下java垃圾回收机制,将其主要内容大致总结一下: 1.什么是垃圾回收机制 java GC机制(garbage collection,垃圾收集,垃圾回收),是java特有的机制,作为jav ...
- Java的多线程机制系列:不得不提的volatile及指令重排序(happen-before)
一.不得不提的volatile volatile是个很老的关键字,几乎伴随着JDK的诞生而诞生,我们都知道这个关键字,但又不太清楚什么时候会使用它:我们在JDK及开源框架中随处可见这个关键字,但并发专 ...
随机推荐
- iOS 多线程 简单学习NSThread NSOperation GCD
1:首先简单介绍什么叫线程 可并发执行的,拥有最小系统资源,共享进程资源的基本调度单位. 共用堆,自有栈(官方资料说明iOS主线程栈大小为1M,其它线程为512K). 并发执行进度不可控,对非原子操作 ...
- PHP中静态方法和实例化方法的区别
在PHP中类为什么要使用静态方法,有什么好处 不需要实例化?? 可以提高运行效率?? 这是一个经常被时时提出来的问题,很多时候我们以为理解了.懂了,但深究一下,我们却发现并不懂. 方法是我们每天都在写 ...
- csdn博客被删除联系客服恢复
前几天写了篇"如何使用shadowsocksFQ",居然被删除,我很郁闷,客服给我的回答是"影响了客户体验,网安查的严,不能带有FQ的信息出现" 我一直很郁闷, ...
- Android 内存暴减的秘密?!
作者:杨超,腾讯移动客户端开发 工程师 商业转载请联系腾讯WeTest获得授权,非商业转载请注明出处. WeTest 导读 在我这样减少了26.5M Java内存! 一文中内存优化一期已经告一段落, ...
- Mysql中字符集总结
有时候,在Mysql数据库中会经常遇到乱码的问题,现在普遍的做法就是全部强行把编码格式都设置成utf8模式,就可以解决这个问题,以前是知其然,不知其所以然,今天我就稍微研究了下Mysql的字符集. 就 ...
- window.location.hash 使用说明
本文给大家详细汇总了关于window.location.hash的知识点,属性以及用法等等,非常的实用,并附上了例子,有需要的小伙伴可以参考下. location是javascript里边管理地址 ...
- PHP重要知识点
1 获取文件名或目录路径 getcwd() :显示是 在哪个文件里调用此文件 的目录 __DIR__ :当前内容写在哪个文件就显示这个文件目录 __FILE__ : 当前内容写在哪个文件就显示这个文件 ...
- python科学计算_numpy_函数库
1.常规函数与排序 常用统计函数: 求和:sum().均值:mean().标准差:std().方差:var().最小值:min().最大值:max().最大值与最小值之差:ptp().最大值的下标:a ...
- Web开发入门学习笔记
公司web项目终于要启动了,本以为django学习可以在实战中进行,结果最终使用了Drupal框架,好吧,那我们就PHP走起,买了本<细说PHP>,先跟着过一遍Web开发入门. HTTP协 ...
- 80 行代码爬取豆瓣 Top250 电影信息并导出到 CSV 及数据库
一.下载页面并处理 二.提取数据 观察该网站 html 结构 可知该页面下所有电影包含在 ol 标签下.每个 li 标签包含单个电影的内容. 使用 XPath 语句获取该 ol 标签 在 ol 标签中 ...