回归面试题!大家应该都听说过 Java 的 Object#notify 方法吧?那么你真的了解这个方法么?Object#notify 的作用是什么?是随机还是顺序唤醒等待线程呢?
回答重点
notify() 会 顺序(在 hotspot 的实现中)唤醒一个调用 wait 后等待的线程。
扩展知识
notify 相关知识
notify() 是 Object 类中的一个方法,用于唤醒在该对象上等待的一个线程。Java 的多线程机制提供了 wait() 和 notify() 这两个方法来实现线程间的协调与通信。
具体来说,如果一个线程调用了某个对象的 wait() 方法,它就会进入等待状态,等待另一个线程调用同一对象的 notify() 方法来唤醒它。会唤醒一个正在等待该对象监视器的线程。如果有多个线程在同一对象上等待,具体唤醒哪一个线程是由 JVM 实现决定的。
注意!调用 notify() 的线程必须拥有该对象的锁,也就是说,notify() 必须在 synchronized 代码块或方法内部执行。否则,会抛出 IllegalMonitorStateException。
notify() 与 notifyAll() 的区别:
- notify():只唤醒一个等待的线程,如果有多个线程在等待,那么被唤醒的线程是随机选择的。
- notifyAll():唤醒在该对象监视器上等待的所有线程,但是这些被唤醒的线程仍然需要竞争锁,只有一个线程能够获取锁并继续执行。
hotspot notify 源码实现
在 Object 源码中 notify 方法上的注释写到:其唤醒 wait 的线程是任意(arbitrary)的,但是具体还是需要看实现:
目前一般我们的 JVM 实现都是 hotspot,因此我们看下 hotspot 源码的实现:
在调用 wait 的时候,可以看到实际上会调用 AddWaiter 方法,从源码来看就是入队排着:
然后 notify 的时候,实际上让排第一个的线程出队:
所以在 hotspot 中,notify 实际上是顺序唤醒线程的。