notify唤醒单个等待线程,notifyAll唤醒所有等待线程;二者均需在synchronized中调用,配合wait实现线程通信,使用while检查条件防止虚假唤醒,多线程环境下推荐优先使用notifyAll以避免死锁。

在Java中,notify 和 notifyAll 是用于线程间通信的重要方法,它们定义在 Object 类中,通常与 wait 方法配合使用,实现线程的等待与唤醒机制。正确使用这些方法可以有效避免资源竞争和死锁问题。
理解 wait、notify 和 notifyAll 的作用
这三个方法都必须在同步上下文中调用(即 synchronized 块或方法中),因为它们依赖于对象的监视器(monitor)。
- wait():使当前线程释放对象锁并进入等待状态,直到其他线程调用该对象的 notify() 或 notifyAll()。
- notify():唤醒在此对象监视器上等待的一个线程(选择是任意的,由JVM调度决定)。
- notifyAll():唤醒在此对象监视器上等待的所有线程,这些线程将重新竞争获取锁。
何时使用 notify
当你确定只有一个等待线程需要被唤醒时,可以使用 notify。例如生产者-消费者模型中只有一个消费者线程。
示例场景:
synchronized (lock) {
// 唤醒一个正在等待的消费者
if (hasData) {
lock.notify();
}
}
注意:虽然只唤醒一个线程,但如果存在多个等待线程,不能保证唤醒的是哪一个。
立即学习“Java免费学习笔记(深入)”;
免费 盛世企业网站管理系统(SnSee)系统完全免费使用,无任何功能模块使用限制,在使用过程中如遇到相关问题可以去官方论坛参与讨论。开源 系统Web代码完全开源,在您使用过程中可以根据自已实际情况加以调整或修改,完全可以满足您的需求。强大且灵活 独创的多语言功能,可以直接在后台自由设定语言版本,其语言版本不限数量,可根据自已需要进行任意设置;系统各模块可在后台自由设置及开启;强大且适用的后台管理支
何时使用 notifyAll
当多个线程可能因不同条件在同一个对象上等待时,应使用 notifyAll,确保符合条件的线程有机会执行。
- 多个消费者和生产者同时运行时。
- 等待的线程可能等待不同的条件(比如队列非空、队列未满等)。
使用 notifyAll 可以避免“信号丢失”或“唤醒错线程”导致的死锁或阻塞。
实际代码示例:简单生产者消费者模型
public class SharedResource {
private String data;
private boolean hasData = false;
public synchronized void consume() throws InterruptedException {
while (!hasData) {
this.wait(); // 等待数据产生
}
System.out.println("消费数据: " + data);
hasData = false;
this.notifyAll(); // 唤醒所有等待线程(包括生产者)
}
public synchronized void produce(String data) throws InterruptedException {
while (hasData) {
this.wait(); // 等待消费完成
}
this.data = data;
hasData = true;
System.out.println("生产数据: " + data);
this.notifyAll(); // 唤醒消费者
}
}
这里使用 while 而不是 if 检查条件,防止虚假唤醒;使用 notifyAll 确保在多线程环境下正确唤醒。
关键注意事项
- 必须在 synchronized 块中调用 wait、notify、notifyAll,否则会抛出 IllegalMonitorStateException。
- 使用 while 循环检查等待条件,避免虚假唤醒问题。
- 优先考虑使用 notifyAll,除非你非常确定只有一个线程在等待。
- 唤醒后线程需重新竞争锁,不会立即执行。









