0

0

在Java中如何实现生产者消费者模型_Java经典并发模型实现解析

P粉602998670

P粉602998670

发布时间:2026-01-21 09:17:03

|

965人浏览过

|

来源于php中文网

原创

Java实现生产者消费者模型应优先使用BlockingQueue,因其天然线程安全且避免唤醒丢失、虚假唤醒等问题;需高级控制时才用ReentrantLock+Condition。

在java中如何实现生产者消费者模型_java经典并发模型实现解析

Java 中实现生产者消费者模型,核心不是手写 wait/notify,而是优先用 java.util.concurrent 包里现成的线程安全组件——它们已解决唤醒丢失、虚假唤醒、锁粒度、公平性等底层问题。

BlockingQueue 实现最简可靠版本

这是最推荐的起点:避免自己管理锁和条件变量,BlockingQueueput()take() 方法天然阻塞且线程安全。

  • ArrayBlockingQueue:固定容量、基于数组、可选公平策略,适合容量明确、注重吞吐或响应公平性的场景
  • LinkedBlockingQueue:默认无界(实际是 Integer.MAX_VALUE),注意 OOM 风险;有界构造时性能略低于 ArrayBlockingQueue
  • SynchronousQueue:不存储元素,每个 put() 必须等待配对 take(),适合“直传”型任务交接,常用于线程池的 DirectHandoff
class Producer implements Runnable {
    private final BlockingQueue<String> queue;

    Producer(BlockingQueue<String> queue) {
        this.queue = queue;
    }

    @Override
    public void run() {
        try {
            for (int i = 0; i < 5; i++) {
                String item = "item-" + i;
                System.out.println("Produced: " + item);
                queue.put(item); // 自动阻塞直到有空间
                Thread.sleep(100);
            }
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }
    }
}

class Consumer implements Runnable {
    private final BlockingQueue<String> queue;

    Consumer(BlockingQueue<String> queue) {
        this.queue = queue;
    }

    @Override
    public void run() {
        try {
            while (true) {
                String item = queue.take(); // 自动阻塞直到有元素
                System.out.println("Consumed: " + item);
                if (item.equals("item-4")) break; // 简单退出条件
                Thread.sleep(150);
            }
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }
    }
}

为什么不用 wait()/notify() 手写?

除非教学或特殊调度逻辑,否则手写极易出错:

  • 必须在 synchronized 块内调用,否则抛 IllegalMonitorStateException
  • notify() 可能唤醒错误线程(比如多个生产者+多个消费者共用一把锁),应优先用 notifyAll()
  • 必须用 while 而非 if 检查条件,否则遭遇虚假唤醒(spurious wakeup)会直接跳过判断导致逻辑崩溃
  • 无法控制唤醒顺序,容易造成线程饥饿(如生产者一直抢到锁,消费者永远等不到)

需要自定义逻辑时:用 ReentrantLock + Condition

当需分离「生产就绪」和「消费就绪」两个等待队列,或需尝试获取、超时获取、中断响应等高级行为时,才考虑该组合。

PathFinder
PathFinder

AI驱动的销售漏斗分析工具

下载

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

  • 一个 Lock 对应多个 Condition:例如 notFullnotEmpty,避免 notifyAll() 唤醒无关线程
  • awaitNanos(long)awaitUntil(Date) 支持超时,比 wait(long) 更灵活
  • lockInterruptibly() 让阻塞中的线程能响应中断,比 synchronized 更可控
class BoundedBuffer<T> {
    private final ReentrantLock lock = new ReentrantLock();
    private final Condition notFull = lock.newCondition();
    private final Condition notEmpty = lock.newCondition();
    private final Object[] items;
    private int putIndex, takeIndex, count;

    BoundedBuffer(int capacity) {
        this.items = new Object[capacity];
    }

    public void put(T x) throws InterruptedException {
        lock.lockInterruptibly();
        try {
            while (count == items.length)
                notFull.await(); // 等待非满
            items[putIndex] = x;
            if (++putIndex == items.length) putIndex = 0;
            ++count;
            notEmpty.signal(); // 唤醒一个消费者
        } finally {
            lock.unlock();
        }
    }

    @SuppressWarnings("unchecked")
    public T take() throws InterruptedException {
        lock.lockInterruptibly();
        try {
            while (count == 0)
                notEmpty.await(); // 等待非空
            Object x = items[takeIndex];
            items[takeIndex] = null;
            if (++takeIndex == items.length) takeIndex = 0;
            --count;
            notFull.signal(); // 唤醒一个生产者
            return (T) x;
        } finally {
            lock.unlock();
        }
    }
}

真正难的从来不是写出能跑的代码,而是让边界条件(如中断、容量耗尽、多线程竞争、JVM 优化重排序)下的行为完全可预期。用好 BlockingQueue 就规避了 90% 的坑;真要手写,ReentrantLock + Condition 是唯一值得投入精力的底层路径。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

847

2023.08.22

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

107

2023.09.25

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

765

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

377

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

32

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

29

2026.01.21

C# 多线程与异步编程
C# 多线程与异步编程

本专题深入讲解 C# 中多线程与异步编程的核心概念与实战技巧,包括线程池管理、Task 类的使用、async/await 异步编程模式、并发控制与线程同步、死锁与竞态条件的解决方案。通过实际项目,帮助开发者掌握 如何在 C# 中构建高并发、低延迟的异步系统,提升应用性能和响应速度。

103

2026.02.06

Python异步编程与Asyncio高并发应用实践
Python异步编程与Asyncio高并发应用实践

本专题围绕 Python 异步编程模型展开,深入讲解 Asyncio 框架的核心原理与应用实践。内容包括事件循环机制、协程任务调度、异步 IO 处理以及并发任务管理策略。通过构建高并发网络请求与异步数据处理案例,帮助开发者掌握 Python 在高并发场景中的高效开发方法,并提升系统资源利用率与整体运行性能。

37

2026.03.12

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

136

2026.03.11

热门下载

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

精品课程

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

共23课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.2万人学习

Java 教程
Java 教程

共578课时 | 81.5万人学习

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

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