生活中,我们常遇到需要等待的场景,例如去银行办事,在没轮到自己之前需要一直等待,但是如果需要自己每隔几秒钟就去柜台前看看状况,那肯定是种非常低效和令人恼火的体验。而实际的情况是,接待员会让您拿个号,说"请稍等一会"(wait); 当排到时,语言和大屏幕会提示"请XXX号到N号柜台办理"(notify)。

wait/notify机制也正是处理这样的场景:线程继续执行需要等待某个条件的变化,这个条件由另一个任务改变,如果一直空循环检查条件变化,是一种不良的CPU使用方式,这时候可以调用wait()将任务挂起,在其他线程调用了notify()或notifyAll()时,任务被唤醒并检查条件的变化。

这个过程中,锁的持有发生了变化。介绍wait/notify最常用的例子是生产者和消费者,设想你去饭馆吃饭,叫来服务员说,把我的宫保鸡丁端上来吧。这时候你获得了服务员的锁,在解决你的事情前,服务员不能去做别的事。(同一时间,厨师可能已经做好了宫保鸡丁,等服务员来端,但是服务员在和你说话,厨师束手无策(等待锁)。)服务员没有宫保鸡丁,只能对你说:您稍等一下,我去厨房催催。服务员调用了wait()方法,你只好释放锁,服务员回到厨房,厨师怒气冲冲的喊(获得锁),宫保鸡丁好了,端走。

下面用程序演示这一场景:

  1. public class Waiter {
  2. private String dishes = null;
  3. public synchronized String getDishes() {
  4. System.out.printf("顾客获得服务员锁%n");
  5. while(this.dishes == null) {
  6. try {
  7. System.out.printf("顾客取菜,没有菜...顾客线程等待(释放锁)%n");
  8. wait();
  9. } catch(InterruptedException ex) {
  10. ex.printStackTrace();
  11. }
  12. }
  13. String d = this.dishes;
  14. System.out.printf("顾客取走: %s%n", this.dishes);
  15. this.dishes = null;
  16. notifyAll();
  17. System.out.printf("服务员通知正在等待的线程%n");
  18. return d;
  19. }
  20. public synchronized void setDishes(String dishes) {
  21. System.out.printf("厨师获得服务员锁%n");
  22. while(this.dishes != null) {
  23. try {
  24. System.out.printf("厨师交菜,服务员已经端了另一份菜...厨师线程等待(释放锁)%n");
  25. wait();
  26. } catch(InterruptedException ex) {
  27. ex.printStackTrace();
  28. }
  29. }
  30. this.dishes = dishes;
  31. System.out.printf("厨师交菜: %s%n", this.dishes);
  32. notifyAll();
  33. System.out.printf("服务员通知正在等待的线程(顾客)%n");
  34. }
  35. public static void main(String[] args) throws InterruptedException {
  36. Waiter busy = new Waiter();
  37. for(int i = 0; i < 10; i++) {
  38. Thread consumer = new Thread() {
  39. public void run() {
  40. busy.getDishes();
  41. }
  42. };
  43. consumer.start();
  44. }
  45. Thread.sleep(100);
  46. for(int i = 0; i < 10; i++) {
  47. Thread chef = new Thread() {
  48. public void run() {
  49. String dishes = "宫保鸡丁";
  50. busy.setDishes(dishes);
  51. }
  52. };
  53. chef.start();
  54. }
  55. }
  56. }

运行结果:

  1. 顾客获得服务员锁
  2. 顾客取菜,没有菜...顾客线程等待(释放锁)
  3. 厨师获得服务员锁
  4. 厨师交菜: 宫保鸡丁
  5. 服务员通知正在等待的线程(顾客)
  6. 顾客取走: 宫保鸡丁
  7. 服务员通知正在等待的线程(厨师)

下面来说明notifyAll的作用。

修改下代码,把厨师和顾客都增加到10个

  1. public static void main(String[] args) {
  2. Waiter busy = new Waiter();
  3. for(int i = 0; i < 10; i++) {
  4. Thread consumer = new Thread() {
  5. public void run() {
  6. busy.getDishes();
  7. }
  8. };
  9. consumer.start();
  10. }
  11. for(int i = 0; i < 10; i++) {
  12. Thread chef = new Thread() {
  13. public void run() {
  14. String dishes = "宫保鸡丁";
  15. busy.setDishes(dishes);
  16. }
  17. };
  18. chef.start();
  19. }
  20. }

执行后会发现程序会陷入永久的等待无法结束,这是因为notify()方法只唤醒众多等待的线程中的一个,拿到菜后本应唤醒顾客取走,但是有可能随机唤醒了另一个等待的厨师,没有顾客能取走服务员手中的菜,这时候程序就无法继续下去了。

解决的方法有两种:

1 把notify()改成notifyAll(),唤醒所有等待的线程

2 使用Java.util.concurrent库中的Condition,把等待的线程分为厨师和顾客两个集合,代码如下:

  1. public class ConditionWaiter {
  2. private String dishes = null;
  3. private Lock lock = new ReentrantLock();
  4. private Condition conConsumer = lock.newCondition();
  5. private Condition conChef = lock.newCondition();
  6. public String getDishes() {
  7. try {
  8. lock.lock();
  9. System.out.printf("顾客获得服务员锁%n");
  10. while(this.dishes == null) {
  11. try {
  12. System.out.printf("顾客取菜,没有菜...顾客线程等待%n");
  13. conConsumer.await();
  14. } catch(InterruptedException ex) {
  15. ex.printStackTrace();
  16. }
  17. }
  18. String d = this.dishes;
  19. System.out.printf("顾客取走:%s%n", this.dishes);
  20. this.dishes = null;
  21. conChef.signal();
  22. System.out.printf("服务员通知正在等待的线程(厨师)%n");
  23. return d;
  24. } finally {
  25. lock.unlock();
  26. }
  27. }
  28. public void setDishes(String dishes) {
  29. try {
  30. lock.lock();
  31. System.out.printf("厨师获得服务员锁%n");
  32. while(this.dishes != null) {
  33. try {
  34. System.out.printf("厨师交菜,服务员已经端了另一份菜...厨师线程等待%n");
  35. conChef.await();
  36. } catch(InterruptedException ex) {
  37. ex.printStackTrace();
  38. }
  39. }
  40. this.dishes = dishes;
  41. System.out.printf("厨师交菜:%s%n", this.dishes);
  42. conConsumer.signal();
  43. System.out.printf("服务员通知正在等待的线程(顾客)%n");
  44. } finally {
  45. lock.unlock();
  46. }
  47. }
  48. public static void main(String[] args) throws InterruptedException {
  49. ConditionWaiter busy = new ConditionWaiter();
  50. for(int i = 0; i < 10; i++) {
  51. Thread consumer = new Thread() {
  52. public void run() {
  53. busy.getDishes();
  54. }
  55. };
  56. consumer.start();
  57. }
  58. Thread.sleep(100);
  59. for(int i = 0; i < 10; i++) {
  60. Thread chef = new Thread() {
  61. public void run() {
  62. String dishes = "宫保鸡丁";
  63. busy.setDishes(dishes);
  64. }
  65. };
  66. chef.start();
  67. }
  68. }
  69. }

事实上,wait/notify机制编程模型复杂也运行低效,通常我们应该采取更高级的类库实现类似场景。以下代码是使用BlockingQueue实现线程协作的示例:

    1. public class BlockingQueueWaiter {
    2. static BlockingQueue<String> queue = new ArrayBlockingQueue<>(1);
    3. public static void main(String[] args) throws InterruptedException {
    4. for(int i = 0; i < 10; i++) {
    5. Thread consumer = new Thread() {
    6. public void run() {
    7. String dishes;
    8. try {
    9. System.out.printf("顾客尝试取菜%n");
    10. dishes = queue.take();
    11. System.out.printf("顾客取走:%s%n", dishes);
    12. } catch (InterruptedException e) {
    13. e.printStackTrace();
    14. }
    15. }
    16. };
    17. consumer.start();
    18. }
    19. Thread.sleep(100);
    20. for(int i = 0; i < 10; i++) {
    21. Thread chef = new Thread() {
    22. public void run() {
    23. String dishes = "宫保鸡丁";
    24. try {
    25. System.out.printf("厨师尝试交菜%n");
    26. queue.put(dishes);
    27. System.out.printf("厨师交菜:%s%n", dishes);
    28. } catch (InterruptedException e) {
    29. e.printStackTrace();
    30. }
    31. }
    32. };
    33. chef.start();
    34. }
    35. }
    36. }

通俗的解释JAVA wait/notify机制的更多相关文章

  1. QWaitCondition(和Java的Notify机制非常相像)

    QT通过三种形式提供了对线程的支持.它们分别是,一.平台无关的线程类,二.线程安全的事件投递,三.跨线程的信号-槽连接.这使得开发轻巧的多线程Qt程序更为容易,并能充分利用多处理器机器的优势.多线程编 ...

  2. 为什么JAVA要提供 wait/notify 机制?是为了避免轮询带来的性能损失

    wait/notify  机制是为了避免轮询带来的性能损失. 为了说清道理,我们用“图书馆借书”这个经典例子来作解释. 一本书同时只能借给一个人.现在有一本书,图书馆已经把这本书借了张三. 在简单的s ...

  3. C# 实现java中 wiat/notify机制

    最近在学习java,看到wiat/notify机制实现线程通信,由于平时工作用的C#,赶紧用C#方式实现一个demo. Java 代码: import java.util.ArrayList; imp ...

  4. Java的多线程机制系列:(一)总述及基础概念

    前言 这一系列多线程的文章,一方面是个人对Java现有的多线程机制的学习和记录,另一方面是希望能给不熟悉Java多线程机制.或有一定基础但理解还不够深的读者一个比较全面的介绍,旨在使读者对Java的多 ...

  5. Java 类反射机制分析

    Java 类反射机制分析 一.反射的概念及在Java中的类反射 反射主要是指程序可以访问.检测和修改它本身状态或行为的一种能力.在计算机科学领域,反射是一类应用,它们能够自描述和自控制.这类应用通过某 ...

  6. JAVA 初识类加载机制 第13节

    JAVA 初识类加载机制 第13节 从这章开始,我们就进入虚拟机类加载机制的学习了.那么什么是类加载呢?当我们写完一个Java类的时候,并不是直接就可以运行的,它还要编译成.class文件,再由虚拟机 ...

  7. 沉淀再出发:再谈java的多线程机制

    沉淀再出发:再谈java的多线程机制 一.前言 自从我们学习了操作系统之后,对于其中的线程和进程就有了非常深刻的理解,但是,我们可能在C,C++语言之中尝试过这些机制,并且做过相应的实验,但是对于ja ...

  8. java垃圾回收机制学习总结

    最近学习了一下java垃圾回收机制,将其主要内容大致总结一下: 1.什么是垃圾回收机制 java GC机制(garbage collection,垃圾收集,垃圾回收),是java特有的机制,作为jav ...

  9. Java的多线程机制系列:不得不提的volatile及指令重排序(happen-before)

    一.不得不提的volatile volatile是个很老的关键字,几乎伴随着JDK的诞生而诞生,我们都知道这个关键字,但又不太清楚什么时候会使用它:我们在JDK及开源框架中随处可见这个关键字,但并发专 ...

随机推荐

  1. 前端MVC Vue2学习总结(三)——模板语法、过滤器、计算属性、观察者、Class 与 Style 绑定

    Vue.js 使用了基于 HTML 的模版语法,允许开发者声明式地将 DOM 绑定至底层 Vue 实例的数据.所有 Vue.js 的模板都是合法的 HTML ,所以能被遵循规范的浏览器和 HTML 解 ...

  2. LINUX:alias命令详解

    发现目前安装的g++并没有开启选项 -std=c++11,无法使用c++11的新标准及其中的列表初始化.搜索后得到解决方法:键入:alias  g++="g++ -std=c++11&quo ...

  3. Ionic2 cordova angular2 打包到Android apk环境搭建

    一.前言 前段时间,公司有个APP项目需要支持不同平台,于是采用了Ionic2 + cordova + angular2,在搭建环境过程中遇到了不少问题,刚好最近有时间整理出来. 二.开发环境搭建 参 ...

  4. ArcGIS API for JavaScript 4.2学习笔记[24] 【IdentifyTask类】的使用(结合IdentifyParameters类)(第七章完结)

    好吧,我都要吐了. 接连三个例子都是类似的套路,使用某个查询参数类的实例,结合对应的Task类,对返回值进行取值.显示. 这个例子是Identify识别,使用了TileLayer这种图层,数据来自Se ...

  5. centos7 yum 安装 redis

    //从中国科学技术大学开源镜像站 wget http://mirrors.ustc.edu.cn/epel/7/x86_64/Packages/e/epel-release-7-11.noarch.r ...

  6. unity demo之坦克攻击

    先展示一下成果吧,节后第一天上班简直困爆了,所以一定要动下脑子搞点事情. 分析: 1.涉及到的游戏对象有:坦克,摄像机,场景素材(包含灯光),子弹 2.坦克具有的功能:移动,旋转,发射子弹,记录生命值 ...

  7. SCOI 2010 序列操作

    题目描述 lxhgww最近收到了一个01序列,序列里面包含了n个数,这些数要么是0,要么是1,现在对于这个序列有五种变换操作和询问操作: 0 a b 把[a, b]区间内的所有数全变成0 1 a b ...

  8. 房上的猫:StringBuffer类

    一.使用StringBuffer类 StringBuffer类位于java.lang包中,是String类的增强类 步骤:  1.声明StringBuffer对象并初始化 StringBuffer s ...

  9. 微信小程序开发之普通链接二维码

    本文主要介绍扫普通链接二维码打开小程序, 详情请看官方文档https://mp.weixin.qq.com/debug/wxadoc/introduction/qrcode.html 配置普通链接二维 ...

  10. Git知识总览(二) git常用命令概览

    上篇博客我们从 git clone 和 git status 两个命令开始,引出了一系列的git操作命令, 请参见:<Git知识总览(一) 从 git clone 和 git status 谈起 ...