一个线程修改一个对象的值,而另一个线程则感知到了变化,然后进行相应的操作,这就是wait()、notify()和notifyAll()方法的本质。具体体现到方法上则是这样的:一个线程A调用了对象obj的wait方法进入到等待状态,而另一个线程调用了对象obj的notify()或者notifyAll()方法,线程A收到通知后从对象obj的wait方法返回,继续执行后面的操作。
可以看到以上两个线程通过对象obj进行操作,而wait和notify/notifyAll的关系就像开关信号一样,用来完成等待方和通知方之间的交互工作。
下面的代码演示了这个过程:分别创建一个等待线程和一个通知线程,前者检查flag的值是否为false,如果符合要求就进行后续的操作,否则在lock上等待。后者在睡眠一段时间后对lock进行通知,等待线程这样就可以从wait方法返回了
package com.rhwayfun.concurrency;import java.text.DateFormat;import java.text.SimpleDateFormat;import java.util.Date;/** * Created by rhwayfun on 16-4-2. */public class WaitNotifyThread { //条件是否满足的标志 private static boolean flag = true; //对象的监视器锁 private static Object lock = new Object(); //日期格式化器 private static DateFormat format = new SimpleDateFormat("HH:mm:ss"); public static void main(String[] args){ Thread waitThread = new Thread(new WaitThread(),"WaitThread"); waitThread.start(); SleepUtil.second(1); Thread notifyThread = new Thread(new NotifyThread(),"NotifyThread"); notifyThread.start(); } /** * 等待线程 */ private static class WaitThread implements Runnable{ public void run() { //加锁,持有对象的监视器锁 synchronized (lock){ //只有成功获取对象的监视器才能进入这里 //当条件不满足的时候,继续wait,直到某个线程执行了通知 //并且释放了lock的监视器(简单来说就是锁)才能从wait //方法返回 while (flag){ try { System.out.println(Thread.currentThread().getName() + " flag is true,waiting at " + format.format(new Date())); lock.wait(); } catch (InterruptedException e) { e.printStackTrace(); } } //条件满足,继续工作 System.out.println(Thread.currentThread().getName() + " flag is false,running at " + format.format(new Date())); } } } /** * 通知线程 */ private static class NotifyThread implements Runnable{ public void run() { synchronized (lock){ //获取lock锁,然后执行通知,通知的时候不会释放lock锁 //只有当前线程退出了lock后,waitThread才有可能从wait返回 System.out.println(Thread.currentThread().getName() + " holds lock. Notify waitThread at " + format.format(new Date())); lock.notifyAll(); flag = false; SleepUtil.second(5); } //再次加锁 synchronized (lock){ System.out.println(Thread.currentThread().getName() + " holds lock again. NotifyThread will sleep at " + format.format(new Date())); SleepUtil.second(5); } } }}
以上代码的输出结果为:
其实使用wait、notify/notifyAll很简单,但是仍然需要注意以下几点:
- 使用wait()、notify()和notifyAll()时需要首先对调用对象加锁
- 调用wait()方法后,线程状态会从RUNNING变为WAITING,并将当线程加入到lock对象的等待队列中
- 调用notify()或者notifyAll()方法后,等待在lock对象的等待队列的线程不会马上从wait()方法返回,必须要等到调用notify()或者notifyAll()方法的线程将lock锁释放,等待线程才有机会从等待队列返回。这里只是有机会,因为锁释放后,等待线程会出现竞争,只有竞争到该锁的线程才会从wait()方法返回,其他的线程只能继续等待
- notify()方法将等待队列中的一个线程移到lock对象的同步队列,notifyAll()方法则是将等待队列中所有线程移到lock对象的同步队列,被移动的线程的状态由WAITING变为BLOCKED
- wait()方法上等待锁,可以通过wait(long timeout)设置等待的超时时间
上一篇文章还有正确恢复线程的问题需要解决,因为通过使用wait()、notify()和notifyAll()可以很好恢复与挂起线程,下面是改进的代码:
package com.rhwayfun.concurrency;import java.text.DateFormat;import java.text.SimpleDateFormat;import java.util.Date;import java.util.concurrent.TimeUnit;/** * Created by rhwayfun on 16-4-2. */public class SafeResumeAndSuspendThread { private static DateFormat format = new SimpleDateFormat("HH:mm:ss"); //对象锁 private static Object lock = new Object(); public static void main(String[] args) throws InterruptedException { Runner r = new Runner(); Thread runThread = new Thread(r,"CountThread"); runThread.start(); //主线程休眠一会,让CountThread有机会执行 TimeUnit.SECONDS.sleep(2); for (int i = 0; i < 3; i++){ //让线程挂起 r.suspendRequest(); //让计数线程挂起两秒 TimeUnit.SECONDS.sleep(2); //看看i的值 System.out.println("after suspend, i = " + r.getValue()); //恢复线程的执行 r.resumeRequest(); //线程休眠一会 TimeUnit.SECONDS.sleep(1); } //退出程序 System.exit(0); } /** * 该线程是一个计数线程 */ private static class Runner implements Runnable{ //变量i private volatile long i; //是否继续运行的标志 //这里使用volatile关键字可以保证多线程并发访问该变量的时候 //其他线程都可以感知到该变量值的变化。这样所有线程都会从共享 //内存中取值 private volatile boolean suspendFlag; public void run() { try { suspendFlag = false; i = 0; work(); } catch (InterruptedException e) { e.printStackTrace(); } } private void work() throws InterruptedException { while (true){ //只有当线程挂起的时候才会执行这段代码 waitWhileSuspend(); i++; System.out.println("calling work method, i = " + i); //只有当线程挂起的时候才会执行这段代码 waitWhileSuspend(); //休眠1秒 TimeUnit.SECONDS.sleep(1); } } /** * 忙等待 * @throws InterruptedException */ private void waitWhileSuspend() throws InterruptedException { /*while (suspendFlag){ TimeUnit.SECONDS.sleep(1); }*/ /** * 等待通知的方式才是最佳选择 */ synchronized (lock){ while (suspendFlag){ System.out.println(Thread.currentThread().getName() + " suspend at " + format.format(new Date())); lock.wait(); } } } //让线程终止的方法 public void resumeRequest(){ synchronized (lock){ try { suspendFlag = false; System.out.print("after call resumeRequest method, i = " + getValue() + ". "); lock.notifyAll(); TimeUnit.SECONDS.sleep(1); } catch (InterruptedException e) { e.printStackTrace(); } } } public void suspendRequest(){ suspendFlag = true; System.out.print("after call suspendRequest method, i = " + getValue() + ". "); } public long getValue(){ return i; } }}
代码的执行结果如下:
可以看到不管是挂起还是恢复,得到的结果都是正确的,在使用等待/通知机制实现的时候,需要注意必须使用同一个lock对象作为两个线程沟通的桥梁,由于synchronized关键字的可重入性(这点后面还会提到),保证了整个程序的正常执行。
总结:正确挂起和恢复线程的方法是使用boolean变量做为标志位,能够在合适的时间和位置正确恢复与挂起线程。