0

0

在Java里notify和notifyAll如何选择_Java线程唤醒策略说明

P粉602998670

P粉602998670

发布时间:2026-01-17 13:36:17

|

359人浏览过

|

来源于php中文网

原创

notify只唤醒一个线程且不可控,易导致虚假唤醒或死锁;notifyAll唤醒所有等待线程并重检条件,更安全通用;实际开发中应优先使用java.util.concurrent工具类。

在java里notify和notifyall如何选择_java线程唤醒策略说明

notify 只唤醒一个线程,但不保证唤醒谁

Java 中 notify() 会从等待队列(wait set)中随机选一个线程唤醒,具体选哪个由 JVM 实现决定,**无法预测、不可控**。这意味着:如果多个线程在等同一个条件,而你只调用 notify(),很可能唤醒的是那个条件仍不满足的线程——它醒来后检查条件失败,又调回 wait(),造成虚假唤醒循环或死锁风险。

常见错误现象:

  • 程序卡住,部分线程永远不继续执行
  • 并发逻辑正确但偶发失败,尤其在高负载或不同 JVM 版本下行为不一致

适用场景极少,仅当满足以下全部条件时才可考虑:

  • 等待队列中**最多只有一个线程**在合理状态下等待(例如单生产者-单消费者且严格串行)
  • 被唤醒线程**无需重新校验条件**(几乎不存在)
  • 你完全掌控所有可能调用 wait() 的路径,且能证明不会出现“唤醒错人”

notifyAll 是更安全、更通用的选择

notifyAll() 唤醒所有在该对象上等待的线程,让它们竞争锁并各自重新判断条件是否成立。虽然看起来“浪费”,但它消除了因唤醒顺序不确定导致的隐藏 bug。

立即学习Java免费学习笔记(深入)”;

性能影响其实被高估了:

  • 现代 JVM 对 notifyAll() 做了优化,未获得锁的线程不会真正“运行”,只是从 wait set 移入 entry set
  • 真正开销来自后续的条件重判和可能的再次 wait,而这本来就是正确同步所必需的
  • 相比死锁或活锁,这点开销几乎可以忽略

典型正确写法(守卫条件必须用 while 而非 if):

synchronized (lock) {
    while (!conditionMet) {
        lock.wait();
    }
    // 处理逻辑
}

对应的通知端也应是:

Bing图像创建器
Bing图像创建器

必应出品基于DALL·E的AI绘图工具

下载
synchronized (lock) {
    conditionMet = true;
    lock.notifyAll(); // 不是 notify()
}

什么时候真该用 notify?几乎没有

文档和教科书常提“notify 用于已知只有一个线程在等”,但现实中这个“已知”极难保证。比如:

  • 线程池中复用线程,旧 wait 状态可能残留
  • 异常分支没清理等待状态,导致多个线程意外进入 wait
  • 子类重写了 wait 相关逻辑,扩大了等待范围

就连 JDK 自身的 java.util.concurrent 包里,也基本不用 notify() —— ArrayBlockingQueueLinkedBlockingQueue 全部使用 notifyAll()

唯一勉强算合规的场景:你写了一个私有、封闭、无继承、无异常穿透的工具类,且单元测试穷举了所有线程调度组合,确认永远只有一个线程在 wait。即便如此,换成 notifyAll() 也不影响功能,还更易维护。

替代方案:优先用 java.util.concurrent 工具类

手写 wait/notify 容易出错,JDK 提供了更高层次的抽象:

  • Condition 配合 ReentrantLock,支持多条件队列,比 Object.wait 更清晰
  • CountDownLatchCyclicBarrierSemaphore 表达明确的同步意图
  • BlockingQueue 替代手动实现生产者-消费者等待逻辑

例如,替换传统 wait/notify 的阻塞队列操作:

BlockingQueue queue = new LinkedBlockingQueue<>(10);
// 生产者
queue.put("item"); // 自动阻塞,无需 wait/notify
// 消费者
String item = queue.take(); // 自动阻塞,无需 wait/notify

这些类内部确实用了 notifyAll() 或更精细的唤醒机制,但封装后你不再需要直面选择难题。

真正容易被忽略的点:即使你坚持用 wait/notify,也要始终配合 while 循环检查条件,并确保所有修改条件的代码都在同一把锁下完成——否则 notifyAll() 也救不了逻辑错误。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

834

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

739

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

735

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

399

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

41

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.6万人学习

C# 教程
C# 教程

共94课时 | 6.8万人学习

Java 教程
Java 教程

共578课时 | 46.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号