Object#notify 有个小坑?多新鲜啊

2024-08-29 12:33:45 浏览数 (2)

回归面试题!大家应该都听说过 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 实际上是顺序唤醒线程的。

0 人点赞