package thread; /** * 15.为Thread撰写两个子类,其中一个的run()在启动后取得第二个Thread object reference, * 然后调用wait()。另一个子类的run()在过了数秒之后调用notifyAll(), * 唤醒第一个线程,使第一个线程可以印出消息。(40分) * * @since 2010-9-21 * @start 8:10 * @end 2010-9-22 5:25 * */ public class Fifty_ThreadInvoke { public static void main(String[] args){ ThreadInvoke ti=new ThreadInvoke(); ti.init(); ti.start(); } } class ThreadInvoke{ Thread t1=null; Thread t2=null; public void init(){ t1=new Thread(){ public void run(){ System.out.println("thread 1 invoke thread.start()"); t2.start(); synchronized(t1){ try { this.wait(); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("thread one waken !!"); } } }; t2=new Thread(){ public void run(){ synchronized(t1){ System.out.println("thread 2 is sleeping ..."); try{ this.sleep(3000); } catch(Exception e){} System.out.println("thread 2 wake and notifyAll"); t1.notifyAll(); } } }; } public void start(){ t1.start(); } } /** * 首先,调用一个Object的wait与notify/notifyAll的时候,必须保证调用代码对该Object是同步的,也就是说必须在作用等同于synchronized(obj){......}的内部才能够去调用obj的wait 与notify/notifyAll三个方法,否则就会报错: java.lang.IllegalMonitorStateException: current thread not owner 在调用wait的时候,线程自动释放其占有的对象锁,同时不会去申请对象锁。当线程被唤醒的时候,它才再次获得了去获得对象锁的权利。 所以,notify与notifyAll没有太多的区别,只是notify仅唤醒一个线程并允许它去获得锁,notifyAll是唤醒所有等待这个对象的线程并允许它们去获得对象锁,只要是在synchronied块中的代码,没有对象锁是寸步难行的。其实唤醒一个线程就是重新允许这个线程去获得对象锁并向下运行。 顺便说一下notifyall,虽然是对每个wait的对象都调用一次notify,但是这个还是有顺序的,每个对象都保存这一个等待对象链,调用的顺序就是这个链的顺序。其实启动等待对象链中各个线程的也是一个线程,在具体应用的时候,需要注意一下。 * */