当前位置: 代码网 > it编程>编程语言>Java > 彻底理解Java线程通信wait / notify(原理 + 实战)

彻底理解Java线程通信wait / notify(原理 + 实战)

2026年03月26日 Java 我要评论
前言在 java 多线程开发中,wait()、notify()和notifyall()是最经典但也最容易写错的一套线程通信机制。它们用于解决:当线程因为“条件不满足”无法继续执

前言

在 java 多线程开发中,wait()notify() 和 notifyall() 是最经典但也最容易写错的一套线程通信机制。
它们用于解决:当线程因为“条件不满足”无法继续执行时,如何安全地等待并在条件改变后继续执行。

这篇文章将从原理、流程、示例到最佳实践,把这三兄弟一次讲透。

一、为什么需要 wait / notify?(最根本的问题)

在多线程场景中,经常会出现:

  • 消费者想消费,但仓库空了
  • 生产者想生产,但仓库满了
  • 主线程需要等待子线程准备好某个结果
  • 条件未满足前,线程无法继续往下执行

如果用 while(true) 死循环检查:

while (item == 0) {
    // 忙等(busy waiting),疯狂消耗 cpu
}

这会导致:

  • cpu 空转
  • 性能极差
  • 线程争抢激烈

于是,java 提供了 基于锁对象的条件等待机制

当条件不满足时,线程主动挂起自己,并释放锁;当条件满足时,被其他线程唤醒继续执行。

这就是 wait / notify

二、wait / notify 的正确理解(核心概念)

1. wait() 的本质动作

当线程执行:

lock.wait();

它会做 三件事

  1. 当前线程暂停执行(挂起)
  2. 释放 lock 的那把锁(让其他线程有机会修改状态)
  3. 把自己加入 lock 对象的“等待队列(wait set)”

线程会一直睡在那里,直到某个线程执行了:

  • lock.notify() 或 lock.notifyall()

并且:

  • 🔥 唤醒的线程必须重新抢 lock 的锁
  • 抢到锁之后,才能继续执行 wait 后面的代码

2. notify() / notifyall() 做了什么?

两个方法都必须在同步代码块中调用:

synchronized(lock) {
    lock.notify();
}

否则会抛:

illegalmonitorstateexception

notify() 作用:

  • 从 lock 的等待队列中 随机唤醒 1 个线程

  • 但唤醒 ≠ 立即执行

  • 被唤醒线程要在当前线程释放锁后才能抢锁执行

notifyall() 作用:

  • 唤醒等待队列中的所有线程

  • 所有线程一起去抢锁,成功者继续执行

实际开发中一般推荐 notifyall():更安全,避免唤醒错误线程导致死锁。

三、完整流程图解(非常关键)

假设线程 a 想消费商品,但仓库空了:

线程 a 做的事:

  1. synchronized(lock) → 拿到锁

  2. 判断仓库是否为空 → 是空的

  3. 调用 wait()

    • a 挂起

    • a 释放锁

    • a 进入等待队列

线程 b(生产者)做的事:

  1. synchronized(lock) → 拿到锁

  2. 生产商品

  3. 调用 notify() 或 notifyall():唤醒 a

  4. b 退出 synchronized → 释放锁

最后:

  • a 被唤醒

  • 抢到锁后,继续执行 wait() 下一行的代码

四、生产者消费者完整 demo(最经典示例)

class depot {
    private int item = 0;
    private final object lock = new object();
    public void produce() throws interruptedexception {
        synchronized (lock) {
            while (item == 1) {   // 仓库满 → 等待
                lock.wait();
            }
            item = 1;
            system.out.println("生产了一个商品");
            lock.notifyall();     // 通知消费者
        }
    }
    public void consume() throws interruptedexception {
        synchronized (lock) {
            while (item == 0) {   // 仓库空 → 等待
                lock.wait();
            }
            item = 0;
            system.out.println("消费了一个商品");
            lock.notifyall();     // 通知生产者
        }
    }
}

为什么要用while而不是 if?

因为:

  • java 的 wait 存在 虚假唤醒(spurious wakeups)

  • 被唤醒后必须重新判断条件

这是 jdk 官方文档明确要求的。

五、wait 和 sleep 有什么区别?(面试必考)

特性waitsleep
属于谁?objectthread
是否释放锁?释放不释放
是否必须在 synchronized 中?必须不需要
唤醒方式notify/notifyall到点自动醒

一句话总结:

sleep 是让线程睡觉
wait 是让线程到等待队列里等别人叫醒

六、常见错误理解(90% 的人会踩坑)

1. 认为 notify 会立即让对方执行

错误!
notify 只是“叫醒”,对方必须等待 当前线程释放锁 后才能执行。

2. 不用 while,用 if 判断条件

会导致唤醒后条件不满足却继续执行 → 出 bug

3. 以为 wait 是“做完事情后休息”

完全相反!

wait 是因为做不下去,不是因为做完了。

4. wait 和 notify 不在同一个锁对象上

比如:

a.wait();
b.notify();

永远唤不醒。

七、总结(建议背下来)

wait 的本质:

线程因为条件不满足 → 挂起自己 → 释放锁 → 进入等待队列 → 等别人唤醒。

notify / notifyall 的本质:

条件改变 → 唤醒等待队列里的线程,让它们重新抢锁继续执行。

wait / notify 是同步代码里的“条件等待机制”,不是休眠机制。

八、这套机制与 rxjava 背压有什么关系?

其实本质一样:

  • rxjava 背压:下游忙 → 上游暂停、等待或限速

  • wait/notify:条件不满足 → 当前线程暂停等待其他线程改变条件

都是:

生产速度与消费速度不一致时的流量控制(flow control)思想。

总结 

到此这篇关于java线程通信wait/notify的文章就介绍到这了,更多相关java线程通信wait/notify内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!

(0)

相关文章:

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论

验证码:
Copyright © 2017-2026  代码网 保留所有权利. 粤ICP备2024248653号
站长QQ:2386932994 | 联系邮箱:2386932994@qq.com