Java中如何实现生产者消费者模式 详解wait/notify机制实现方式

生产者消费者模式通过协调生产者和消费者对共享缓冲区的访问,实现多线程协作。1. 使用wait()/notifyall()机制:当缓冲区满时生产者等待,空时消费者等待,通过notifyall()唤醒线程避免死锁;2. 选择合适的阻塞队列:如arrayblockingqueue(有界队列适合稳定场景)、linkedblockingqueue(适合速度差异大场景)、priorityblockingqueue(优先级处理)、delayqueue(延迟任务)和synchronousqueue(传递性场景);3. 其他实现方式:包括使用blockingqueue简化代码、reentrantlock与condition提供更灵活控制。不同方法适用于不同需求,blockingqueue适合简单实现,reentrantlock适合复杂控制,而wait/notify是基础理解方式。

Java中如何实现生产者消费者模式 详解wait/notify机制实现方式

生产者消费者模式是一种经典的多线程协作模式,它巧妙地平衡了生产速度和消费速度,避免了资源浪费和数据丢失。在Java中,我们可以利用wait()notify()/notifyAll()方法来实现这一模式。简单来说,生产者负责生产数据并放入共享缓冲区,消费者负责从缓冲区取出数据进行消费。

Java中如何实现生产者消费者模式 详解wait/notify机制实现方式

解决方案

Java中如何实现生产者消费者模式 详解wait/notify机制实现方式

实现生产者消费者模式的核心在于协调生产者和消费者对共享缓冲区的访问。以下是一个简单的Java实现:

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

import java.util.LinkedList;import java.util.Queue;public class ProducerConsumer {    private static final int CAPACITY = 5;    private final Queue buffer = new LinkedList();    private final Object lock = new Object();    class Producer implements Runnable {        @Override        public void run() {            int i = 0;            while (true) {                synchronized (lock) {                    try {                        while (buffer.size() == CAPACITY) {                            System.out.println("Buffer is full, Producer is waiting");                            lock.wait(); // 缓冲区满,生产者等待                        }                        buffer.offer(i);                        System.out.println("Produced: " + i);                        i++;                        lock.notifyAll(); // 通知消费者                        Thread.sleep((long) (Math.random() * 100)); // 模拟生产时间                    } catch (InterruptedException e) {                        Thread.currentThread().interrupt();                    }                }            }        }    }    class Consumer implements Runnable {        @Override        public void run() {            while (true) {                synchronized (lock) {                    try {                        while (buffer.isEmpty()) {                            System.out.println("Buffer is empty, Consumer is waiting");                            lock.wait(); // 缓冲区空,消费者等待                        }                        int value = buffer.poll();                        System.out.println("Consumed: " + value);                        lock.notifyAll(); // 通知生产者                        Thread.sleep((long) (Math.random() * 200)); // 模拟消费时间                    } catch (InterruptedException e) {                        Thread.currentThread().interrupt();                    }                }            }        }    }    public static void main(String[] args) {        ProducerConsumer pc = new ProducerConsumer();        new Thread(pc.new Producer()).start();        new Thread(pc.new Consumer()).start();        new Thread(pc.new Consumer()).start(); // 多个消费者    }}

这段代码中,buffer 是共享缓冲区,lock 是用于同步的锁对象。ProducerConsumer 分别是生产者和消费者线程。关键在于 wait()notifyAll() 的使用:

Java中如何实现生产者消费者模式 详解wait/notify机制实现方式wait():当缓冲区满(生产者)或空(消费者)时,线程调用 wait() 方法释放锁并进入等待状态。notifyAll():当生产者生产了新的数据或消费者消费了数据后,调用 notifyAll() 方法唤醒所有等待的线程,让它们重新竞争锁。

为什么使用 notifyAll() 而不是 notify()

虽然 notify() 也能唤醒一个等待的线程,但在生产者消费者模式中,使用 notifyAll() 更安全。 考虑以下情况:如果只有一个消费者线程在等待,notify() 可以唤醒它。但是,如果有多个消费者线程都在等待,而 notify() 唤醒的恰好也是一个消费者线程,那么这个线程可能会发现缓冲区仍然是空的,然后再次进入等待状态。这样就可能导致死锁。notifyAll() 确保所有等待的线程都有机会被唤醒并检查条件,从而避免死锁。

如何选择合适的阻塞队列?

Java提供了多种阻塞队列,例如 ArrayBlockingQueueLinkedBlockingQueuePriorityBlockingQueueDelayQueueSynchronousQueue 等。选择合适的阻塞队列取决于具体的应用场景:

ArrayBlockingQueue: 基于数组实现的有界阻塞队列,需要指定容量。适合于生产者和消费者速度相对稳定的场景。LinkedBlockingQueue: 基于链表实现的阻塞队列,可以选择有界或无界。无界队列可能导致内存溢出,需要谨慎使用。适合于生产者和消费者速度差异较大的场景。PriorityBlockingQueue: 具有优先级的无界阻塞队列。元素必须实现 Comparable 接口。适合于需要按照优先级处理任务的场景。DelayQueue: 延迟队列,元素需要在指定的延迟时间后才能被消费。元素必须实现 Delayed 接口。适合于实现定时任务的场景。SynchronousQueue: 同步队列,每个插入操作必须等待一个相应的移除操作,反之亦然。可以看作是一个容量为 0 的阻塞队列。适合于传递性场景,例如线程池。

选择阻塞队列时,需要考虑以下因素:

是否有界: 有界队列可以防止内存溢出,但可能导致生产者阻塞。无界队列则没有这个限制,但需要注意内存消耗。是否需要优先级: 如果需要按照优先级处理任务,则需要选择 PriorityBlockingQueue是否需要延迟: 如果需要在指定的延迟时间后才能消费元素,则需要选择 DelayQueue并发性能: 不同阻塞队列的并发性能可能有所不同,需要根据实际情况进行选择。

除了wait/notify,还有哪些实现生产者消费者模式的方式?

除了 wait()/notify() 机制,Java还提供了其他方式来实现生产者消费者模式,例如:

BlockingQueue: Java并发包 java.util.concurrent 提供的阻塞队列接口。它简化了生产者消费者模式的实现,无需手动管理锁和等待/通知机制。例如上面的代码可以改为使用 ArrayBlockingQueue

import java.util.concurrent.ArrayBlockingQueue;import java.util.concurrent.BlockingQueue;public class ProducerConsumerBlockingQueue {    private static final int CAPACITY = 5;    private final BlockingQueue buffer = new ArrayBlockingQueue(CAPACITY);    class Producer implements Runnable {        @Override        public void run() {            int i = 0;            while (true) {                try {                    buffer.put(i); // 阻塞直到队列有空间                    System.out.println("Produced: " + i);                    i++;                    Thread.sleep((long) (Math.random() * 100));                } catch (InterruptedException e) {                    Thread.currentThread().interrupt();                }            }        }    }    class Consumer implements Runnable {        @Override        public void run() {            while (true) {                try {                    int value = buffer.take(); // 阻塞直到队列有元素                    System.out.println("Consumed: " + value);                    Thread.sleep((long) (Math.random() * 200));                } catch (InterruptedException e) {                    Thread.currentThread().interrupt();                }            }        }    }    public static void main(String[] args) {        ProducerConsumerBlockingQueue pc = new ProducerConsumerBlockingQueue();        new Thread(pc.new Producer()).start();        new Thread(pc.new Consumer()).start();        new Thread(pc.new Consumer()).start();    }}

使用 BlockingQueue 可以大大简化代码,并提高可读性。put() 方法在队列满时会阻塞,take() 方法在队列空时会阻塞,无需手动处理等待和通知。

ReentrantLockCondition: ReentrantLock 提供了比 synchronized 更灵活的锁机制,Condition 则提供了比 wait()/notify() 更精细的线程等待和通知机制。 可以实现更复杂的同步逻辑。

import java.util.LinkedList;import java.util.Queue;import java.util.concurrent.locks.Condition;import java.util.concurrent.locks.Lock;import java.util.concurrent.locks.ReentrantLock;public class ProducerConsumerReentrantLock {    private static final int CAPACITY = 5;    private final Queue buffer = new LinkedList();    private final Lock lock = new ReentrantLock();    private final Condition notFull = lock.newCondition();    private final Condition notEmpty = lock.newCondition();    class Producer implements Runnable {        @Override        public void run() {            int i = 0;            while (true) {                lock.lock();                try {                    while (buffer.size() == CAPACITY) {                        System.out.println("Buffer is full, Producer is waiting");                        notFull.await(); // 缓冲区满,生产者等待                    }                    buffer.offer(i);                    System.out.println("Produced: " + i);                    i++;                    notEmpty.signalAll(); // 通知消费者                    Thread.sleep((long) (Math.random() * 100));                } catch (InterruptedException e) {                    Thread.currentThread().interrupt();                } finally {                    lock.unlock();                }            }        }    }    class Consumer implements Runnable {        @Override        public void run() {            while (true) {                lock.lock();                try {                    while (buffer.isEmpty()) {                        System.out.println("Buffer is empty, Consumer is waiting");                        notEmpty.await(); // 缓冲区空,消费者等待                    }                    int value = buffer.poll();                    System.out.println("Consumed: " + value);                    notFull.signalAll(); // 通知生产者                    Thread.sleep((long) (Math.random() * 200));                } catch (InterruptedException e) {                    Thread.currentThread().interrupt();                } finally {                    lock.unlock();                }            }        }    }    public static void main(String[] args) {        ProducerConsumerReentrantLock pc = new ProducerConsumerReentrantLock();        new Thread(pc.new Producer()).start();        new Thread(pc.new Consumer()).start();        new Thread(pc.new Consumer()).start();    }}

在这个例子中,notFullnotEmpty 是两个 Condition 对象,分别用于生产者和消费者的等待和通知。await() 方法类似于 wait()signalAll() 方法类似于 notifyAll()。 使用 ReentrantLockCondition 可以实现更细粒度的控制,例如可以只唤醒特定的生产者或消费者。

选择哪种方式取决于具体的需求。 如果只是简单的生产者消费者模式,使用 BlockingQueue 是最简单和推荐的方式。 如果需要更灵活的控制,可以使用 ReentrantLockCondition。 而 wait()/notify() 机制则是最基础的方式,理解它可以帮助我们更好地理解并发编程的原理。

以上就是Java中如何实现生产者消费者模式 详解wait/notify机制实现方式的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月5日 17:00:36
下一篇 2025年11月10日 02:23:06

相关推荐

  • js怎样实现网格布局动画 js网格动画的5种交互效果

    javascript实现网格布局动画的核心是结合css grid布局与dom操作,通过动态修改样式属性触发视觉效果。1. 创建css grid容器并定义行列结构;2. 使用javascript操控网格项的样式或借助gsap、anime.js等库实现动画;3. 通过事件监听实现交互效果如悬停放大、颜色…

    2025年12月5日 web前端
    000
  • 电脑主机装机后系统性能检测与调优方法,确保硬件发挥最大潜力

    装完电脑主机后,系统性能检测和调优至关重要。1. 首先进行基本检测,使用cpu-z、gpu-z、crystaldiskinfo等工具确认硬件是否被正确识别;2. 进入bios优化设置,开启xmp/expo配置文件、关闭节能模式、调整风扇曲线;3. 系统层面更新最新驱动、关闭不必要的启动项、设置高性能…

    2025年12月5日 游戏教程
    000
  • 2699元起?华为nova 15系列售价曝光 预计10月发布

    近日,有数码博主透露了华为即将推出的nova 15系列的定价详情。消息称,该系列将延续此前的产品定价思路,标准版起售价或定为2699元,pro版为3499元,ultra版则为4199元。这一价格与2025年5月发布的nova 14系列完全相同——后者同样以2699元起步,pro版3499元,ultr…

    2025年12月5日
    000
  • 如何在Laravel中创建自定义命令

    在laravel中创建自定义命令的步骤如下:1. 使用php artisan make:command mycustomcommand生成命令骨架;2. 在mycustomcommand.php中设置$signature定义命令名、参数和选项,如my:greet {name} {–upp…

    2025年12月5日
    000
  • VSCode怎么更改鼠标颜色_VSCode自定义鼠标指针颜色与光标样式设置教程

    VSCode无法更改系统鼠标指针颜色,但可自定义编辑器内文本光标样式、颜色及行为。通过修改settings.json文件,可设置光标样式(如line、block、underline)、宽度、闪烁方式(如blink、smooth、solid)、颜色(via workbench.colorCustomi…

    2025年12月5日
    000
  • 如何在Laravel中实现搜索功能

    在laravel中实现搜索功能最直接的方式是使用数据库like查询,适用于小规模应用;若需处理大规模数据或复杂搜索逻辑,则应引入laravel scout配合algolia或meilisearch等专业搜索服务。1. 对于简单场景,通过表单提交、路由定义和控制器中的like查询即可实现基础搜索功能;…

    2025年12月5日
    000
  • PHP如何调用Scala代码 通过JVM桥接调用Scala程序的方法

    通过jvm桥接,php可调用scala代码,但需中间工具。具体步骤如下:1. 将scala代码编译为jar包,并确保类和方法为public;2. 部署javabridge到支持servlet的web服务器(如tomcat);3. 在php中配置java.inc并设置classpath以加载jar包;…

    2025年12月5日 后端开发
    000
  • Gartner:2024年全球半导体营收6559亿美元,英伟达首登榜首

    2024年全球半导体市场强劲增长,总收入达6559亿美元,同比增长21%。gartner最新数据显示,市场格局发生显著变化,英伟达凭借ai基础设施建设和数据中心gpu需求的强劲增长,首次超越三星电子和英特尔,荣登全球第一大半导体厂商宝座。 ☞☞☞AI 智能聊天, 问答助手, AI 智能搜索, 免费无…

    2025年12月5日
    100
  • 如何在Laravel中使用事件和监听器

    事件和监听器是laravel中实现松耦合的关键机制。1. 定义事件类如userregistered,封装发生的“事情”;2. 创建监听器如sendwelcomeemail,处理事件触发后的操作,并可异步执行;3. 在eventserviceprovider中注册事件与监听器的映射关系;4. 使用ev…

    2025年12月5日
    000
  • Java中Spock的用法 详解测试框架

    spock是一个针对java和groovy应用程序的测试框架,其核心优势在于简洁性、强大功能与易读语法,尤其适合行为驱动开发(bdd)。1. spock通过groovy语言的动态特性提升测试代码的表现力;2. 它整合了junit、mockito、hamcrest等工具的优点,简化测试流程;3. 核心…

    2025年12月5日 java
    200
  • 京东怎么看商品价格走势?价格波动的原因是什么?五大核心原因详解!

    在京东购物时,你是否注意到同一件商品今天的价格与昨天大不相同?从智能手机到家用电器,从日常用品到高端奢侈品,京东平台上的价格变动无时无刻不在影响着消费者的购买决策。这种波动不仅反映了商家灵活的定价策略,也揭示了市场供需关系的实时变化。依托大数据与人工智能技术,京东商家能够迅速响应市场动向调整售价。而…

    2025年12月5日
    000
  • OpenAI创始人力挺脑机接口初创公司 对抗马斯克Neuralink

    8月13日,据媒体报道,openai及其ceo山姆·奥特曼(sam altman)正计划支持一家名为merge labs的脑机接口初创企业,此举被视为对特斯拉掌门人埃隆·马斯克(elon musk)旗下neuralink的正面挑战,再次点燃两位科技领袖之间的长期竞争。 知情人士透露,Merge La…

    2025年12月5日
    200
  • Composer licenses命令是做什么的_项目依赖许可证信息查询

    composer licenses 命令可列出项目所有依赖的许可证信息,帮助开发者识别开源组件及其合规风险。它读取 composer.lock 文件,展示每个依赖包的名称、版本和许可证类型(如 MIT、Apache-2.0),便于发现潜在法律问题,尤其对商业项目至关重要。该命令是管理许可证的起点,但…

    2025年12月5日
    000
  • 清理PHPCMS数据库冗余数据的操作步骤

    识别并清理phpcms数据库冗余数据需从历史版本、无效附件、重复统计、缓存、垃圾评论等入手。1.识别冗余:通过查看大表结构定位冗余来源;2.备份数据库:使用mysqldump或系统工具备份;3.清理历史版本:编写sql删除旧版本;4.清理无效附件:用php脚本校验文件存在性后删除;5.合并重复统计:…

    2025年12月5日 后端开发
    000
  • 如何在Laravel中使用模型工厂

    laravel中创建和使用模型工厂的步骤为:首先通过artisan命令创建工厂文件,如php artisan make:factory postfactory –model=post;其次在生成的工厂文件中定义definition()方法,设置模型字段及对应假数据,例如使用faker库生…

    2025年12月5日
    100
  • Java中如何生成XML 详解DOM方式创建XML文档

    使用dom方式创建xml文档的步骤如下:1. 创建documentbuilderfactory对象;2. 创建documentbuilder对象;3. 创建document对象;4. 创建根元素并添加到document对象;5. 创建子元素和文本节点;6. 将元素逐级添加到dom树;7. 使用tra…

    2025年12月5日 java
    000
  • 如何生成验证码?GD库图形处理教程

    生成验证码的核心在于服务器端图像处理技术,常用php的gd库实现。其步骤包括:1.创建画布并定义尺寸;2.分配背景、文字及干扰颜色;3.生成随机字符并存入session;4.绘制文字(可用imagettftext增加自然扭曲);5.添加干扰元素如点、线;6.输出图片并销毁资源。传统验证码仍有价值在于…

    2025年12月5日 后端开发
    000
  • Java中Servlet的生命周期 图解Servlet从初始化到销毁的过程

    servlet的生命周期主要包括加载、初始化、处理请求和服务终止四个阶段。1.加载阶段:servlet容器在首次接收请求或启动时加载servlet类;2.初始化阶段:容器创建实例并调用init()方法,该方法仅执行一次,用于读取配置、建立数据库连接等初始化操作;3.处理请求阶段:每次请求到达时,容器…

    2025年12月5日 java
    000
  • 如何在Laravel中创建服务提供者

    服务提供者在laravel中是应用启动和核心功能注册的枢纽。1. 创建服务提供者可通过artisan命令生成文件;2. 在config/app.php中添加服务提供者类以注册它;3. register()方法用于绑定服务到容器,保持简洁仅做绑定操作;4. boot()方法用于执行启动逻辑,如注册事件…

    2025年12月5日
    100
  • 0.198 超低风阻 AI 智能纯电轿车极越 07 首秀北京车展

    展览会随着新车发布、新技术展露,每一次的大型车展都是购车用户和汽车爱好者的最佳去处。本届北京车展将于4月25日开幕,极越作为高端智能汽车品牌,将携带 ” 最美 7 系 ” 极越 07 和 ai 智能纯电 suv 极越 01 双双亮相。 本届北京车展,极越以” 极越…

    2025年12月5日 硬件教程
    100

发表回复

登录后才能评论
关注微信