public static void main(String[] args) throws Exception { final Object lock=""; Thread t1=new Thread(){ public void run(){ try { System.out.println("t1 wait begin"); synchronized (lock) { lock.wait(); } System.out.println("t1 wait end"); } catch (InterruptedException e) { e.printStackTrace(); } } }; t1.start(); Thread.sleep(5000); System.out.println("main lock begin"); synchronized (lock) { lock.notify(); Thread.sleep(10000); System.out.println("main wait begin "); lock.wait(); System.out.println("main wait end"); } System.out.println("main lock end"); t1.join(); System.out.println("process exist"); }
输出:
t1 wait begin main lock begin main wait begin t1 wait end
描述:
线程t1获取lock对象的监视器后,马上调用lock的wait方法,放弃了lock的监视器,
主线程获取lock对象的监视器后,调用lock的notify方法唤醒等待lock的监视器的线程(这里是t1)
之后睡眠10秒,然后调用lock的wait方法放弃lock的监视器。
结论:
1、wait,notify,notifyAll必须在当前线程获得监视器时才能调用,即这些方法必须在同步块中才能调用
2、t1线程,wait方法执行时,t1线程放弃lock对象的监视器,t1线程阻塞,导致同步块代码未执行完。
3、主线程中notify方法执行时,主线程唤醒在等待lock对象监视器的线程(随机的,t1线程只是被标记为可获取监视 器,但实际未获取,详情请看4),另外此方法并不阻塞而是立即返回。
4、主线程notify方法执行后,主线程仍然具有lock对象的监视器,而t1线程仍处于阻塞状态(虽然已经被主线程notify 了), 这点可以看代码,主线程睡眠10秒,而t1线程仍阻塞。只有当主线程调用wait方法,放弃lock的监视器后,
t1线程才真正获得监视器,接着主线程阻塞,而t1线程继续执行同步块中未执行的代码
5、因没有任何线程调用notify方法呼唤主线程了,所以主线程一直阻塞