【Java并发编程】之十:使用wait/notify/notifyAll实现线程间通信的几点重要说明

原创 2013年12月09日 22:09:26

转载请注明出处:http://blog.csdn.net/ns_code/article/details/17225469


   在Java中,可以通过配合调用Object对象的wait()方法和notify()方法或notifyAll()方法来实现线程间的通信。在线程中调用wait()方法,将阻塞等待其他线程的通知(其他线程调用notify()方法或notifyAll()方法),在线程中调用notify()方法或notifyAll()方法,将通知其他线程从wait()方法处返回。


      Object是所有类的超类,它有5个方法组成了等待/通知机制的核心:notify()、notifyAll()、wait()、wait(long)和wait(long,int)。在Java中,所有的类都从Object继承而来,因此,所有的类都拥有这些共有方法可供使用。而且,由于他们都被声明为final,因此在子类中不能覆写任何一个方法。

 

     这里详细说明一下各个方法在使用中需要注意的几点:

 

      1、wait()

      public final void wait()  throws InterruptedException,IllegalMonitorStateException

     该方法用来将当前线程置入休眠状态,直到接到通知或被中断为止。在调用wait()之前,线程必须要获得该对象的对象级别锁,即只能在同步方法或同步块中调用wait()方法。进入wait()方法后,当前线程释放锁。在从wait()返回前,线程与其他线程竞争重新获得锁。如果调用wait()时,没有持有适当的锁,则抛出IllegalMonitorStateException,它是RuntimeException的一个子类,因此,不需要try-catch结构。

 

     2、notify()

     public final native void notify() throws IllegalMonitorStateException

        该方法也要在同步方法或同步块中调用,即在调用前,线程也必须要获得该对象的对象级别锁,的如果调用notify()时没有持有适当的锁,也会抛出IllegalMonitorStateException。

     该方法用来通知那些可能等待该对象的对象锁的其他线程。如果有多个线程等待,则线程规划器任意挑选出其中一个wait()状态的线程来发出通知,并使它等待获取该对象的对象锁(notify后,当前线程不会马上释放该对象锁,wait所在的线程并不能马上获取该对象锁,要等到程序退出synchronized代码块后,当前线程才会释放锁,wait所在的线程也才可以获取该对象锁),但不惊动其他同样在等待被该对象notify的线程们。当第一个获得了该对象锁的wait线程运行完毕以后,它会释放掉该对象锁,此时如果该对象没有再次使用notify语句,则即便该对象已经空闲,其他wait状态等待的线程由于没有得到该对象的通知,会继续阻塞在wait状态,直到这个对象发出一个notify或notifyAll。这里需要注意:它们等待的是被notify或notifyAll,而不是锁。这与下面的notifyAll()方法执行后的情况不同。 

 

     3、notifyAll()

     public final native void notifyAll() throws IllegalMonitorStateException

      该方法与notify()方法的工作方式相同,重要的一点差异是:

      notifyAll使所有原来在该对象上wait的线程统统退出wait的状态(即全部被唤醒,不再等待notify或notifyAll,但由于此时还没有获取到该对象锁,因此还不能继续往下执行),变成等待获取该对象上的锁,一旦该对象锁被释放(notifyAll线程退出调用了notifyAll的synchronized代码块的时候),他们就会去竞争。如果其中一个线程获得了该对象锁,它就会继续往下执行,在它退出synchronized代码块,释放锁后,其他的已经被唤醒的线程将会继续竞争获取该锁,一直进行下去,直到所有被唤醒的线程都执行完毕。

 

     4、wait(long)和wait(long,int)

     显然,这两个方法是设置等待超时时间的,后者在超值时间上加上ns,精度也难以达到,因此,该方法很少使用。对于前者,如果在等待线程接到通知或被中断之前,已经超过了指定的毫秒数,则它通过竞争重新获得锁,并从wait(long)返回。另外,需要知道,如果设置了超时时间,当wait()返回时,我们不能确定它是因为接到了通知还是因为超时而返回的,因为wait()方法不会返回任何相关的信息。但一般可以通过设置标志位来判断,在notify之前改变标志位的值,在wait()方法后读取该标志位的值来判断,当然为了保证notify不被遗漏,我们还需要另外一个标志位来循环判断是否调用wait()方法。

 


       深入理解:

   如果线程调用了对象的wait()方法,那么线程便会处于该对象的等待池中,等待池中的线程不会去竞争该对象的锁。

   当有线程调用了对象的notifyAll()方法(唤醒所有wait线程)或notify()方法(只随机唤醒一个wait线程),被唤醒的的线程便会进入该对象的锁池中,锁池中的线程会去竞争该对象锁。

   优先级高的线程竞争到对象锁的概率大,假若某线程没有竞争到该对象锁,它还会留在锁池中,唯有线程再次调用wait()方法,它才会重新回到等待池中。而竞争到对象锁的线程则继续往下执行,直到执行完了synchronized代码块,它会释放掉该对象锁,这时锁池中的线程会继续竞争该对象锁。


 

 

版权声明:本文为博主原创文章,未经博主允许不得转载。

3.1等待/通知机制(wait/notify)

wait/notify 要点 wait() 执行前 在调用wait()之前,必须先要获得对象锁,即只有在同步方法或者同步代码块中调用wait()方法。 执行作用 wait()使进...
  • bb10210521
  • bb10210521
  • 2016年01月18日 15:18
  • 211

一个理解wait()与notify()的例子

下面是我原来在CSDN论坛上看到的一个贴子,涉及到同步,wait(),notify()等概念的理解,我试着根据原来的一些回复和Think in Java上的相关概念将wait()和notify()这两...
  • xm4014
  • xm4014
  • 2003年01月28日 13:47
  • 14159

java线程的wait方法和notify方法的运用

一. java Thread类的几个函数的说明:  1.  setPriority( ): 设置线程的优先权;     2 . yield( ): 暂停线程的执行,给其它具有相同优先权的线程执行的...
  • liranke
  • liranke
  • 2013年10月01日 18:24
  • 36495

wait和notify的理解与使用

1.对于wait()和notify()的理解对于wait()和notify()的理解,还是要从jdk官方文档中开始,在Object类方法中有: void notify() Wakes ...
  • jianiuqi
  • jianiuqi
  • 2016年12月03日 23:09
  • 5801

深入理解Java多线程中的wait(),notify()和sleep()

大家在学习Java的过程中,势必要进行多线程的系统学习,这部分内容知识对于你在工作中的影响是极大的,并且在面试的过程中,这部分知识也是必然会被问到的。既然多线程的知识如此重要,那么我们就不能浅尝辄止。...
  • Kaka534
  • Kaka534
  • 2016年07月07日 10:41
  • 12946

java中线程wait() notify() 的理解

在JAVA中,是没有类似于PV操作、进程互斥等相关的方法的。JAVA的进程同步是通过synchronized()来实现的,需要说明的是,JAVA的synchronized()方法类似于操作系统概念中的...
  • qbz_96
  • qbz_96
  • 2014年10月18日 20:40
  • 413

【BLE】CC2541之主机端获取notify数据

本文以SimpleBLECentral工程和SimpleBLEPeripheral为例,介绍CC2541作为主机时是如何获取从机char4通道的notify数据...
  • feilusia
  • feilusia
  • 2016年02月17日 19:41
  • 5956

Java wait() notify()方法使用实例讲解

1)wait()、notify()和notifyAll()方法是本地方法,并且为final方法,无法被重写。   2)调用某个对象的wait()方法能让当前线程阻塞,并且当前线程必须拥有此对象的mon...
  • lingzhm
  • lingzhm
  • 2015年04月08日 14:54
  • 21966

JAVA线程同步中wait()和notify()简洁例子

搞懂这两个的用法之前,请你务必搞懂线程同步的道理,否则,下面这一大篇你应该是看不懂的。 wait()和notify()一系列的方法,是属于对象的,不是属于线程的。它们用在线程同步时,synchron...
  • superit401
  • superit401
  • 2016年08月19日 20:40
  • 3619

java wait()notify释放锁

1  调用obj.wait()会立即释放锁,,以便其他线程可以执行obj.notify(),但是notify()不会立刻立刻释放sycronized(obj)中的obj锁,必须要等notify()所...
  • a314773862
  • a314773862
  • 2017年01月05日 16:58
  • 1370
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:【Java并发编程】之十:使用wait/notify/notifyAll实现线程间通信的几点重要说明
举报原因:
原因补充:

(最多只允许输入30个字)