首页 > Java > java教程 > 正文

如何在Java中使用notifyAll唤醒线程

P粉602998670
发布: 2025-09-25 08:19:02
原创
384人浏览过
notifyAll()用于唤醒所有等待特定对象监视器的线程,需在synchronized块中调用,配合wait()实现线程协作,如生产者-消费者模型中通过notifyAll()确保多个消费者或生产者被唤醒,避免线程阻塞。

如何在java中使用notifyall唤醒线程

在Java中,notifyAll() 用于唤醒所有正在等待特定对象监视器的线程。它通常与 synchronized 块、wait() 方法配合使用,实现线程间的协作。下面介绍如何正确使用 notifyAll() 来唤醒线程。

1. 理解 wait() 和 notifyAll() 的工作机制

这三个方法都定义在 Object 类中,必须在 synchronized 上下文中调用:

  • wait():使当前线程释放锁并进入等待状态,直到被唤醒。
  • notify():随机唤醒一个等待该对象锁的线程。
  • notifyAll():唤醒所有等待该对象锁的线程,这些线程将重新竞争获取锁。
注意:直接调用这些方法而不加 synchronized 会抛出 IllegalMonitorStateException。

2. 使用步骤和代码示例

以下是一个生产者-消费者模型的简单示例,展示如何用 notifyAll() 正确唤醒多个等待线程:

import java.util.LinkedList;
import java.util.Queue;

public class NotifyAllExample {
    private final Queue<String> queue = new LinkedList<>();
    private final int MAX_SIZE = 3;

    public void produce(String item) throws InterruptedException {
        synchronized (this) {
            while (queue.size() == MAX_SIZE) {
                System.out.println("队列已满,生产者等待...");
                this.wait(); // 释放锁并等待
            }
            queue.add(item);
            System.out.println("生产了: " + item);
            this.notifyAll(); // 唤醒所有等待线程(包括消费者)
        }
    }

    public String consume() throws InterruptedException {
        synchronized (this) {
            while (queue.isEmpty()) {
                System.out.println("队列为空,消费者等待...");
                this.wait(); // 释放锁并等待
            }
            String item = queue.poll();
            System.out.println("消费了: " + item);
            this.notifyAll(); // 唤醒所有等待线程(包括生产者)
            return item;
        }
    }
}
登录后复制

3. 多线程测试 notifyAll 的效果

启动多个生产者和消费者线程,观察 notifyAll 如何唤醒多个等待线程:

如知AI笔记
如知AI笔记

如知笔记——支持markdown的在线笔记,支持ai智能写作、AI搜索,支持DeepseekR1满血大模型

如知AI笔记 27
查看详情 如知AI笔记

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

public class TestNotifyAll {
    public static void main(String[] args) {
        NotifyAllExample example = new NotifyAllExample();

        // 启动多个消费者线程
        for (int i = 1; i <= 3; i++) {
            new Thread(() -> {
                try {
                    while (true) {
                        example.consume();
                        Thread.sleep(2000); // 模拟处理时间
                    }
                } catch (InterruptedException e) {
                    Thread.currentThread().interrupt();
                }
            }, "Consumer-" + i).start();
        }

        // 启动生产者线程
        new Thread(() -> {
            int counter = 1;
            try {
                while (true) {
                    example.produce("item-" + counter++);
                    Thread.sleep(500);
                }
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
            }
        }, "Producer").start();
    }
}
登录后复制

4. 关键注意事项

使用 notifyAll() 时需注意以下几点:

  • 始终在 synchronized 块或方法中调用 wait() 和 notifyAll()。
  • 使用 while 而不是 if 判断等待条件,防止虚假唤醒或条件变化后继续执行。
  • notifyAll() 会唤醒所有等待线程,但它们需要重新竞争锁,只有一个能进入同步块。
  • 适合场景:当多个线程可能等待同一条件时(如多个消费者),notifyAll() 更安全。

基本上就这些。notifyAll() 是实现线程协作的重要工具,合理使用可避免死锁和线程饥饿问题。

以上就是如何在Java中使用notifyAll唤醒线程的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

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