文章目录
1:等待(wait)和通知(notify)的介绍
JDK提供了两个非常重要的接口线程:等待wait()方法和通知notify()方法。这两个方法在Object类中,意味着任何对象都可以调用这两个方法。方法签名如下
public final void wait() throws InterruptedException {
public final native void notify();
当在一个对象实例上调用wait方法后,当前线程就会在这个对象上等待。这是什么意思呢?比如,在线程A中,调用了obj.wait()方法,那么线程A就会停止继续执行,转为等待状态。等待到何时结束呢?线程A会一直等到其他线程调用了obj.notify()方法为止。这时,object对象成为了多个线程之间通信的有效手段
2:等待(wait)和通知(notify)的执行机制
那么waitO方法和notify0方法究竟是如何工作的呢?下图展示了两者的工作过程。如果一个线程调用了object.wait()方法,那么它就会进入object对象的等待队列。这个等待队列中,可能会有多个线程,因为系统运行多个线程同时等待某一个对象。 当objec.notify()方法被调用时,它就会从这个等待队列中随机选择一个线程, 并将其唤醒。这个选择是不公平的,并不是先等待的线程就会优先被选择,这个选择完全是随机的。
除notifyO方法外,Object 对象还有一个类似的notifyAll()方法,它和notif()方法的功能基本一致,不同的是,它会唤醒在这个等待队列中所有等待的线程,而不是随机选择一个。
这里还需要强调一点,Object.waitO方法 并不能随便调用。它必须包含在对应的synchronzied语句中,无论是wait(0方法或者notify0方法都需要首先获得目标对象的一个监视器。下图 显示了wait()方法和notify()方法的工作流程细节。其中T1和T2表示两个线程。TI在正确执行wait方法前,必须获得object对象的监视器。而wait方法在执行后,会释放这个监视器。这样做的目的是使其他等待在object对象上的线程不至于因为T1的休眠而全部无法正常执行。
线程T2在notify方法调用前,也必须获得objet对象的监视器。所幸,此时T1已经释放了这个监视器。因此,T2可以顺利获得object对象的监视器。接着,T2执行了notify方法尝试唤醒一个等待线程,这里假设唤醒了T1。 T1在被唤醒后,要做的第一件事并不是执行后续的代码,而是要尝试重新获得object 对象的监视器,而这个监视器也正是T1在wait方法执行前所持有的那个。如果暂时无法获得,则T1还必须等待这个监视器。当监视器顺利获得后,T1才可以真正意义上继续执行
这里给出一个使用wait和notify方法的简单案例
public class ThreadWaitAndNotify {
static Object object=new Object()