登录
首页 >  文章 >  java教程

Javanotify()唤醒机制详解

时间:2025-12-18 20:21:55 439浏览 收藏

推广推荐
免费电影APP ➜
支持 PC / 移动端,安全直达

文章不知道大家是否熟悉?今天我将给大家介绍《Java notify()唤醒机制解析》,这篇文章主要会讲到等等知识点,如果你在看完本篇文章后,有更好的建议或者发现哪里有问题,希望大家都能积极评论指出,谢谢!希望我们能一起加油进步!

notify()只唤醒一个等待线程,由JVM随机选择,不释放锁,需配合while循环检测条件,否则易引发并发问题。

java notify()的唤醒

Java 中 notify() 的唤醒行为,核心在于它只唤醒一个正在该对象上等待(wait())的线程,且不保证唤醒哪一个——这是由 JVM 线程调度器决定的,无法预测。

notify() 只唤醒一个等待线程

调用 notify() 时,JVM 会从该对象的等待队列(wait set)中随机选取一个线程,将其从 WAITING 状态移出,进入 BLOCKED 状态(等待重新获取对象锁),之后才可能继续执行。

  • 如果多个线程在同一个对象上调用了 wait()notify() 只唤醒其中一个,其余仍保持等待
  • 若此时没有线程在该对象上等待,notify() 不做任何事,也不会报错
  • notify() 必须在同步块或同步方法中调用,否则抛出 IllegalMonitorStateException

notify() 不释放锁,唤醒后需竞争锁

调用 notify() 本身不会释放当前持有的对象锁;被唤醒的线程只有在当前 synchronized 块/方法执行完毕、锁被释放后,才能参与锁竞争。

  • 被唤醒的线程不会立刻执行,而是先排队争抢锁;若锁被其他线程抢先获得,它将继续阻塞在 entry set 中
  • 因此,被 notify() 唤醒 ≠ 立刻恢复运行,中间存在“锁竞争”环节
  • 常见误写:notify(); return; —— 若后续还有关键逻辑未执行完就返回,可能导致状态不一致

与 notifyAll() 的关键区别

当多个等待线程的「唤醒条件不同」时,仅用 notify() 容易导致信号丢失或死锁。

  • 例如:生产者-消费者中,有多个消费者等待“非空”,多个生产者等待“非满”。若只用 notify(),可能唤醒了同类型的线程(如唤醒消费者但缓冲区仍空),造成虚假唤醒或长期挂起
  • notifyAll() 唤醒所有等待线程,让它们各自重新检查条件(推荐配合 while 循环使用),更安全
  • 仅当能严格保证「每次最多只有一个线程满足条件」且「唤醒目标唯一」时,notify() 才可安全使用(如简单的一对一线程协作)

正确使用模式:wait 必须在 while 循环中

无论用 notify() 还是 notifyAll()wait() 都必须放在 while 循环里,不能用 if

  • 原因:存在虚假唤醒(spurious wakeup)和条件变化竞争,线程被唤醒后,原条件可能已不再成立
  • 正确写法:while (!condition) { obj.wait(); }
  • 唤醒后必须再次检查条件,不满足就继续 wait,避免逻辑错误

基本上就这些。notify() 看似简单,但用错容易引发隐蔽的并发问题,关键是理解它的非确定性、不释放锁、以及必须配合循环检测条件这三个要点。

文中关于java,notify()的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《Javanotify()唤醒机制详解》文章吧,也可关注golang学习网公众号了解相关技术文章。

相关阅读
更多>
最新阅读
更多>
课程推荐
更多>