首先,我们知道wait和notify方法是Object类下的,用来让线程放弃锁等待和唤醒,既然与线程相关,为什么没有放在Thread类
下,而是放在了Object类下呢?(当然由于Thread类继承了Object类,所以Thread也可以调用者三个方法)
其实这个问题很简单,由于每个对象都拥有monitor锁,所以让当前线程等待某个对象的锁,当然应该通过这个对象来操作了。而
不是用当前线程来操作,因为当前线程可能会等待多个线程的锁,如果通过线程来操作,就非常复杂了。
上面已经提到,如果调用某个对象的wait()方法,当前线程必须拥有这个对象的monitor(即锁),因此调用wait()方法必须在
同步块或者同步方法中进行(synchronized块或者synchronized方法)。
调用某个对象的wait()方法,相当于让当前线程交出此对象的monitor,然后进入等待状态,等待后续再次获得此对象的锁
(Thread类中的sleep方法使当前线程暂停执行一段时间,从而让其他线程有机会继续执行,但它并不释放对象锁);
notify()方法能够唤醒一个正在等待该对象的monitor的线程,当有多个线程都在等待该对象的monitor的话,则只能随机唤醒
其中一个线程,具体唤醒哪个线程则不得而知。
同样地,调用某个对象的notify()方法,当前线程也必须拥有这个对象的monitor,因此调用notify()方法必须在同步块或者同步
方法中进行(synchronized块或者synchronized方法)。
nofityAll()方法能够唤醒所有正在等待该对象的monitor的线程,这一点与notify()方法是不同的。
这里要注意一点:notify()和notifyAll()方法只是唤醒等待该对象的monitor的线程,并不决定哪个线程能够获取到monitor。
假如有三个线程Thread1、Thread2和Thread3都在等待对象objectA的monitor,此时Thread4拥有对象objectA的monitor,当在
Thread4中调用objectA.notify()方法之后,Thread1、Thread2和Thread3只有一个能被唤醒。注意,被唤醒不等于立刻就获取了
objectA的monitor。假若在Thread4中调用objectA.notifyAll()方法,则Thread1、Thread2和Thread3三个线程都会被唤醒,至于哪
个线程接下来能够获取到objectA的monitor就具体依赖于操作系统的调度了。
上面尤其要注意一点,一个线程被唤醒不代表立即获取了对象的monitor,只有等调用完notify()或者notifyAll()并退出
synchronized块,释放对象锁后,其余线程才可获得锁执行。
总结
- sleep方法只会让线程睡眠,让出CPU,但不会释放占有的锁资源
- sleep方法可以在任何地方执行
- sleep是Thread类的方法
- wait方法不仅会让出CPU使用权,还会释放占用的锁资源,自身进入对象的等待池中
- wait方法只能在synchronize方法或synchronize代码块中执行
- wait是Object类的方法