首页 > Java > java教程 > 正文

在Java中如何使用ReentrantLock结合Condition实现线程通信_Condition线程通知实践

P粉602998670
发布: 2025-11-01 10:18:02
原创
610人浏览过
ReentrantLock结合Condition可实现更灵活的线程协作,通过newCondition()创建多个等待队列,使用await()/signal()替代wait/notify,避免虚假唤醒。示例中生产者消费者模型利用notFull和notEmpty两个Condition精确控制线程等待与唤醒,确保线程安全与高效通信。

在java中如何使用reentrantlock结合condition实现线程通信_condition线程通知实践

在Java中,ReentrantLock 提供了比 synchronized 更灵活的锁机制,而结合 Condition 可以实现更精确的线程等待与通知。相比传统的 wait/notify,Condition 允许一个锁创建多个等待条件,从而实现更复杂的线程协作场景。

Condition 的基本原理

每个 Condition 实例都由 ReentrantLock 创建,通过 lock.newCondition() 获取。它提供了类似 Object 的 wait、notify 和 notifyAll 方法,分别是:

  • await():当前线程释放锁并进入等待状态
  • signal():唤醒一个等待中的线程
  • signalAll():唤醒所有等待中的线程

使用 Condition 能避免虚假唤醒,并支持多个独立的等待队列。

生产者-消费者模型示例

下面是一个典型的生产者-消费者问题,使用 ReentrantLock 和 Condition 实现线程间通信:

AI卡通生成器
AI卡通生成器

免费在线AI卡通图片生成器 | 一键将图片或文本转换成精美卡通形象

AI卡通生成器 51
查看详情 AI卡通生成器

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

import java.util.LinkedList;
import java.util.Queue;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantLock;

public class ProducerConsumerExample {
    private final Queue<Integer> queue = new LinkedList<>();
    private final int MAX_SIZE = 5;

    private final ReentrantLock lock = new ReentrantLock();
    private final Condition notFull = lock.newCondition();
    private final Condition notEmpty = lock.newCondition();

    class Producer implements Runnable {
        @Override
        public void run() {
            for (int i = 1; i <= 10; i++) {
                lock.lock();
                try {
                    while (queue.size() == MAX_SIZE) {
                        System.out.println("队列已满,生产者等待...");
                        notFull.await(); // 等待空间
                    }
                    queue.offer(i);
                    System.out.println("生产: " + i);
                    notEmpty.signal(); // 通知消费者可以消费
                } catch (InterruptedException e) {
                    Thread.currentThread().interrupt();
                } finally {
                    lock.unlock();
                }
            }
        }
    }

    class Consumer implements Runnable {
        @Override
        public void run() {
            for (int i = 0; i < 10; i++) {
                lock.lock();
                try {
                    while (queue.isEmpty()) {
                        System.out.println("队列为空,消费者等待...");
                        notEmpty.await(); // 等待数据
                    }
                    Integer value = queue.poll();
                    System.out.println("消费: " + value);
                    notFull.signal(); // 通知生产者可以生产
                } catch (InterruptedException e) {
                    Thread.currentThread().interrupt();
                } finally {
                    lock.unlock();
                }
            }
        }
    }

    public static void main(String[] args) {
        ProducerConsumerExample example = new ProducerConsumerExample();
        Thread producer = new Thread(example.new Producer());
        Thread consumer = new Thread(example.new Consumer());

        producer.start();
        consumer.start();
    }
}
登录后复制

关键点说明与最佳实践

上述代码展示了 Condition 的典型用法,以下是需要注意的关键细节:

  • 必须在 lock() 和 unlock() 之间调用 await()/signal(),否则会抛出 IllegalMonitorStateException
  • 使用 while 判断条件而非 if,防止虚假唤醒或多个线程被唤醒时状态再次不满足
  • signal() 唤醒至少一个等待线程,但具体唤醒哪个取决于调度策略;若需唤醒全部,使用 signalAll()
  • 多个 Condition 可以实现不同条件下的等待,比如读锁和写锁分离

基本上就这些。ReentrantLock 配合 Condition 不仅提升了线程通信的灵活性,还让逻辑更清晰可控,适合复杂同步场景的开发。掌握这种模式对理解并发编程非常有帮助。

以上就是在Java中如何使用ReentrantLock结合Condition实现线程通信_Condition线程通知实践的详细内容,更多请关注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号