学习笔记-07生产者-消费者模型4种实现方式

在Java中,生产者-消费者模型可以通过多种方式实现。以下是常见的几种实现方法及其代码示例:

  1. **使用 `wait()` 和 `notify()`(基础同步机制)

通过 `synchronized` 块和 `Object` 的等待/唤醒机制实现。

public class WaitNotifyExample {

private final Queue<Integer> queue = new LinkedList<>();

private final int MAX_SIZE = 10;

public void produce() throws InterruptedException {

while (true) {

synchronized (queue) {

while (queue.size() == MAX_SIZE) {

queue.wait(); // 队列满时等待

}

int value = new Random().nextInt(100);

queue.add(value);

System.out.println("生产: " + value);

queue.notifyAll(); // 唤醒消费者

}

Thread.sleep(500);

}

}

public void consume() throws InterruptedException {

while (true) {

synchronized (queue) {

while (queue.isEmpty()) {

queue.wait(); // 队列空时等待

}

int value = queue.poll();

System.out.println("消费: " + value);

queue.notifyAll(); // 唤醒生产者

}

Thread.sleep(1000);

}

}

}

  1. 使用 `BlockingQueue`(线程安全队列)

直接利用 `BlockingQueue` 的阻塞特性简化代码。

import java.util.concurrent.BlockingQueue;

import java.util.concurrent.LinkedBlockingQueue;

public class BlockingQueueExample {

private final BlockingQueue<Integer> queue = new LinkedBlockingQueue<>(10);

public void produce() throws InterruptedException {

while (true) {

int value = new Random().nextInt(100);

queue.put(value); // 队列满时自动阻塞

System.out.println("生产: " + value);

Thread.sleep(500);

}

}

public void consume() throws InterruptedException {

while (true) {

int value = queue.take(); // 队列空时自动阻塞

System.out.println("消费: " + value);

Thread.sleep(1000);

}

}

}

  1. 使用 `Lock` 和 `Condition`(更灵活的锁)

通过显式锁和条件变量实现细粒度控制。

import java.util.concurrent.locks.*;

import java.util.Queue;

import java.util.LinkedList;

public class LockConditionExample {

private final Queue<Integer> queue = new LinkedList<>();

private final int MAX_SIZE = 10;

private final Lock lock = new ReentrantLock();

private final Condition notFull = lock.newCondition();

private final Condition notEmpty = lock.newCondition();

public void produce() throws InterruptedException {

while (true) {

lock.lock();

try {

while (queue.size() == MAX_SIZE) {

notFull.await(); // 等待队列不满

}

int value = new Random().nextInt(100);

queue.add(value);

System.out.println("生产: " + value);

notEmpty.signal(); // 唤醒消费者

} finally {

lock.unlock();

}

Thread.sleep(500);

}

}

public void consume() throws InterruptedException {

while (true) {

lock.lock();

try {

while (queue.isEmpty()) {

notEmpty.await(); // 等待队列不空

}

int value = queue.poll();

System.out.println("消费: " + value);

notFull.signal(); // 唤醒生产者

} finally {

lock.unlock();

}

Thread.sleep(1000);

}

}

}

  1. 使用 `Semaphore`(信号量控制资源)

通过信号量管理可用资源数量。

import java.util.concurrent.Semaphore;

import java.util.Queue;

import java.util.LinkedList;

public class SemaphoreExample {

private final Queue<Integer> queue = new LinkedList<>();

private final int MAX_SIZE = 10;

private final Semaphore semProducer = new Semaphore(MAX_SIZE);

private final Semaphore semConsumer = new Semaphore(0);

private final Object lock = new Object();

public void produce() throws InterruptedException {

while (true) {

semProducer.acquire(); // 获取生产许可

synchronized (lock) {

int value = new Random().nextInt(100);

queue.add(value);

System.out.println("生产: " + value);

}

semConsumer.release(); // 释放消费许可

Thread.sleep(500);

}

}

public void consume() throws InterruptedException {

while (true) {

semConsumer.acquire(); // 获取消费许可

synchronized (lock) {

int value = queue.poll();

System.out.println("消费: " + value);

}

semProducer.release(); // 释放生产许可

Thread.sleep(1000);

}

}

}

总结

以上四种是Java中实现生产者-消费者的主流方式:

  1. `wait()/notify()`:适合基础场景,需手动处理同步。

  2. `BlockingQueue`:代码最简洁,推荐优先使用。

  3. `Lock` + `Condition`:提供更灵活的锁控制。

  4. `Semaphore`:通过资源计数管理同步,需注意线程安全。

根据具体需求(如性能、复杂度、可扩展性)选择合适的方式。

相关推荐
毕设源码-朱学姐16 分钟前
【开题答辩全过程】以 基于JavaWeb的网上家具商城设计与实现为例,包含答辩的问题和答案
java
四维碎片1 小时前
【Qt】UDP跨平台调试工具
qt·学习·udp
好奇龙猫2 小时前
【人工智能学习-AI入试相关题目练习-第十八次】
人工智能·学习
C雨后彩虹2 小时前
CAS与其他并发方案的对比及面试常见问题
java·面试·cas·同步·异步·
程序员辣条2 小时前
AI产品经理:2024年职场发展的新机遇
人工智能·学习·职场和发展·产品经理·大模型学习·大模型入门·大模型教程
wanping158259923412 小时前
AI Agent(学习六-FAISS 持久化到磁盘(重启不丢记忆))
人工智能·学习·faiss
知识分享小能手2 小时前
SQL Server 2019入门学习教程,从入门到精通,SQL Server 2019数据库的操作(2)
数据库·学习·sqlserver
java1234_小锋3 小时前
Java高频面试题:SpringBoot为什么要禁止循环依赖?
java·开发语言·面试
2501_944525543 小时前
Flutter for OpenHarmony 个人理财管理App实战 - 账户详情页面
android·java·开发语言·前端·javascript·flutter
计算机学姐3 小时前
基于SpringBoot的电影点评交流平台【协同过滤推荐算法+数据可视化统计】
java·vue.js·spring boot·spring·信息可视化·echarts·推荐算法