前言:在调用wait方法时,线程补习持有被调用对象的锁(monitor),当调用完wait方法之后,线程就会释放掉该对象的锁(monitor);而在调用Thread类的sleep方法时,线程是不会释放掉对象的锁的;
关于wait与notify和notifyAll方法的总结:
1.当调用wait时,首先需要确保调用了wait方法的线程已经持有了对象的锁。
2.当调用完wait方法后,该线程就会释放掉这个对象的锁,然后进入到等到状态(wait set)锁池?
3.当线程调用了wait后进入到等待状态时,它就可以等待其他线程调用相同对象的notify或notifyAll方法来使得自己被唤醒。
4.一旦这个线程被其他线程唤醒后,该线程就会与其他线程一同开始竞争这个对象的锁(公平竞争);只有当该线程获取到了这个对象的锁后,线程才会继续往下执行。
5.调用wait方法的代码片段需要放在一个synchronized块或者是synchronized方法中,这样才可以确保线程在调用wait方法前就已经获取到了对象的锁。
6.当调用对象的notify方法时,它会随机唤醒该对象等待集合(wait set)中的任意一个线程,当某个线程被唤醒后,它就会与其他线程一同竞争对象的锁。
7.当调用对象的notifyAll方法时,它会唤醒该对象等待集合(wait set)中的所有线程,这些线程被唤醒后,又会开始竞争对象的锁。
8.在某一时刻,只有唯一一个线程可以拥有对象的锁。
锁池与等待池关系图:
PS:每一个Java对象都有一个唯一对应的Monitor对象,Monitor对象就是我们通常所说的锁,Monitor对象是有C++编写和维护的。