wait
该方法用来将当前线程释放锁,并且进入wait pool,此时没有资格争抢锁。
notify():
唤醒在等待该对象同步锁的线程(只唤醒一个,如果有多个在等待),注意的是在调用此方法的时候,并不能确切的唤醒某一个等待状态的线程,而是由JVM确定唤醒哪个线程,而且不是按优先级。
调用任意对象的notify()方法则导致因调用该对象的 wait()方法而阻塞的线程中随机选择的一个解除阻塞(但要等到获得锁后才真正可执行)。
notifyAll():
唤醒所有等待的线程,注意唤醒的是notify之前wait的线程,对于notify之后的wait线程是没有效果的。
什么时候用notify?什么时候notifyAll?
基于上面的原理,notifyAll会叫醒所有线程争抢锁,所以notify性能更高;所以两者均可用的时候,应该使用notify。
a.错用notify导致的假死
当有多个生产者或者消费者的时候,使用的是notify而不是notifyall,那么有可能唤醒的是同类的,即有可能生产者唤醒的是生产者,消费者唤醒的是消费者。
解决办法:使用notifyall
但是如果仅有一个生产者和消费者,此时应该调用notify,互相叫醒即可。但是使用notifyall似乎一样。
如果wait的有多个线程,并且确保notify其中任何一个线程都是ok的,那么就用notify,如一个生产者,多个消费者,生产者在叫醒消费者的场景。
当某个线程调用notifyAll方法后,虽然其他线程被唤醒了,但是该线程依然持有着对象锁,必须等该同步代码块执行完(右大括号结束)后才算正式释放了锁对象,别的线程才有机会获取锁执行。