【JavaEE】阻塞队列,代码模拟实现阻塞队列并应用于生产者消费者模型

了解阻塞队列

阻塞队列是Java中的一种线程安全的队列,通常用于实现生产者-消费者模式。它不仅可以存储数据,还提供了当队列为空或已满时线程的阻塞能力。阻塞队列在多线程环境中非常有用,可以有效地解决线程间的通信和协调问题。

在Java中,阻塞队列的主要实现有:

  1. ArrayBlockingQueue:基于数组结构的阻塞队列,容量有限,按FIFO(先进先出)原则操作。
  2. LinkedBlockingQueue:基于链表结构的阻塞队列,容量可选择无限或指定,具有较高的并发性能。
  3. PriorityBlockingQueue:支持优先级排序的阻塞队列,元素按照自然顺序或提供的比较器排序。
  4. DelayQueue:一个支持延迟排队的阻塞队列,只有在元素到达预定时间后才能被 consumed。

阻塞队列提供了一些重要的方法,如put()take(),用于插入和获取元素。这些方法在队列满或空时会导致调用线程阻塞,从而有效控制线程的执行流程。

总之,阻塞队列是处理多线程编程中任务调度及数据共享的一个重要工具,能提高代码的可读性和可靠性。

代码模拟实现阻塞队列

模拟实现

下面是一个简单的Java实现的阻塞队列示例。这个实现使用了wait()notifyAll()来处理线程的阻塞和唤醒。我们的自定义阻塞队列类 MyBlockingQueue 支持基本的 put 和 take 方法。

import java.util.LinkedList;public class MyBlockingQueue<T> {private final LinkedList<T> queue;private final int capacity;public MyBlockingQueue(int capacity) {this.queue = new LinkedList<>();this.capacity = capacity;}// 添加元素到队列public synchronized void put(T item) throws InterruptedException {while (queue.size() == capacity) {wait(); // 队列满时,等待}queue.add(item);notifyAll(); // 唤醒其他线程}// 从队列中获取元素public synchronized T take() throws InterruptedException {while (queue.isEmpty()) {wait(); // 队列为空时,等待}T item = queue.removeFirst();notifyAll(); // 唤醒其他线程return item;}// 查看队列大小public synchronized int size() {return queue.size();}// 检查队列是否为空public synchronized boolean isEmpty() {return queue.isEmpty();}
}

使用示例

下面是一个使用上述 MyBlockingQueue 的示例,包括生产者和消费者的实现:

public class BlockingQueueExample {public static void main(String[] args) {MyBlockingQueue<Integer> queue = new MyBlockingQueue<>(5); // 创建容量为5的阻塞队列// 生产者线程Thread producer = new Thread(() -> {try {for (int i = 0; i < 10; i++) {queue.put(i);System.out.println("Produced: " + i);Thread.sleep(100); // 模拟生产时间}} catch (InterruptedException e) {Thread.currentThread().interrupt();}});// 消费者线程Thread consumer = new Thread(() -> {try {for (int i = 0; i < 10; i++) {Integer value = queue.take();System.out.println("Consumed: " + value);Thread.sleep(150); // 模拟消费时间}} catch (InterruptedException e) {Thread.currentThread().interrupt();}});producer.start();consumer.start();}
}

总结

上面的代码展示了如何实现一个简单的阻塞队列以及生产者-消费者模式的基本用法。你可以根据需要扩展这个基础实现,例如添加更多的功能和错误处理。

代码解析

下面是对上述代码实现的详细解析,包括 MyBlockingQueue 类和生产者-消费者示例的工作原理。

1. MyBlockingQueue 类

这个类实现了一个基于链表的阻塞队列,包括两个主要的方法 put 和 take,以及一些辅助方法。

属性

  • queue: 使用 LinkedList 来存储队列中的元素。
  • capacity: 指定队列的最大容量。

方法

  • 构造方法:

public MyBlockingQueue(int capacity) {this.queue = new LinkedList<>();this.capacity = capacity;
}

这里我们初始化一个空的链表和指定的容量。

  • put(T item) 方法:

public synchronized void put(T item) throws InterruptedException {while (queue.size() == capacity) {wait(); // 队列满时,等待}queue.add(item);notifyAll(); // 唤醒其他线程
}

使用 synchronized 关键字保证了方法的线程安全性。

如果队列已满,调用 wait() 方法使当前线程等待,直到有空间可以添加新元素。

当队列有空间时,添加元素,然后调用 notifyAll() 方法唤醒所有在调用 wait() 的线程(可能是消费者线程)。

  • take() 方法:
public synchronized T take() throws InterruptedException {while (queue.isEmpty()) {wait(); // 队列为空时,等待}T item = queue.removeFirst();notifyAll(); // 唤醒其他线程return item;
}

同样地,是一个同步方法。

如果队列为空,调用 wait() 使当前线程等待,直到有元素可以取出。

当有元素可取时,移除并返回队列的第一个元素,并调用 notifyAll() 以唤醒其他线程(如生产者线程)。

辅助方法:

  • size(): 返回当前队列的大小。
  • isEmpty(): 检查队列是否为空。

2. 生产者-消费者示例

在 BlockingQueueExample 类中,我们创建了一个阻塞队列,并定义了生产者和消费者线程。

public class BlockingQueueExample {public static void main(String[] args) {MyBlockingQueue<Integer> queue = new MyBlockingQueue<>(5); // 创建容量为5的阻塞队列

这里我们创建了一个最大容量为5的阻塞队列实例。

生产者线程

Thread producer = new Thread(() -> {try {for (int i = 0; i < 10; i++) {queue.put(i);System.out.println("Produced: " + i);Thread.sleep(100); // 模拟生产时间}} catch (InterruptedException e) {Thread.currentThread().interrupt();}
});

生产者线程循环10次,生成数据(0到9)。

每次生成一个数据调用 put 方法加入队列。

如果队列已满,生产者会被阻塞,直到有空间。

消费者线程

Thread consumer = new Thread(() -> {try {for (int i = 0; i < 10; i++) {Integer value = queue.take();System.out.println("Consumed: " + value);Thread.sleep(150); // 模拟消费时间}} catch (InterruptedException e) {Thread.currentThread().interrupt();}
});
  • 消费者线程同样循环10次,从队列中取数据。
  • 如果队列为空,消费者会被阻塞,直到有数据可取。

总结

运行这个代码时,生产者和消费者线程会交替工作,生产数据并放入队列,消费者从队列中取出数据。由于使用了同步和阻塞机制,代码能够安全地在多线程环境中运行。通过 wait() 和 notifyAll() 的使用,有效地管理了线程的阻塞和唤醒,确保了生产者和消费者之间的协调和通信。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/52235.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

卸载nomachine

网上的方法:提示找不到命令 我的方法: step1. 终端输入 sudo find / -name nxserver 2>/dev/null确认 NoMachine 的实际安装路径。你可以使用 find 命令在系统中查找 nxserver 脚本的位置。 找到路径后,你可以使用该路径来卸载 NoMachine。 如下图,紫色框中是我的路径…

Android - lock/unlock bootloader

在执行 adb remount 时高版本经常会提示失败 此时就需要对设备的进行解锁操作。记录两个部分&#xff0c;Google解锁和展锐解锁。 目录 一、Google解锁 二、展锐解锁 三、补充跳过按键检测的方案 一、Google解锁 官网介绍的unlock方法如下&#xff1a;锁定/解锁引导加载程序…

RK3588 技术分享 | 在Android系统中使用NPU实现Yolov5分类检测-迅为电子

随着人工智能和大数据时代的到来&#xff0c;传统嵌入式处理器中的CPU和GPU逐渐无法满足日益增长的深度学习需求。为了应对这一挑战&#xff0c;在一些高端处理器中&#xff0c;NPU&#xff08;神经网络处理单元&#xff09;也被集成到了处理器里。NPU的出现不仅减轻了CPU和GPU…

【Java 优先队列(小顶堆) 分治法 实现合并k个排序链表】

合并k个排序链表 题目&#xff1a;力扣-合并k个排序链表[https://leetcode.cn/problems/vvXgSW/](https://leetcode.cn/problems/vvXgSW/)优先队列&#xff08;小顶堆&#xff09;法代码实现 分治法代码实现 题目&#xff1a;力扣-合并k个排序链表https://leetcode.cn/problems…

Linux基础环境开发工具gcc/g++ make/Makefile git

1.Linux编译器-gcc/g使用 1. 预处理&#xff08;进行宏替换) 预处理功能主要包括宏定义,文件包含,条件编译,去注释等。 预处理指令是以#号开头的代码行。 实例: gcc –E hello.c –o hello.i 选项“-E”,该选项的作用是让 gcc 在预处理结束后停止编译过程。 选项“-o”是指目标…

【Java数据结构】---二叉树OJ

乐观学习&#xff0c;乐观生活&#xff0c;才能不断前进啊&#xff01;&#xff01;&#xff01; 我的主页&#xff1a;optimistic_chen 我的专栏&#xff1a;c语言 &#xff0c;Java 欢迎大家访问~ 创作不易&#xff0c;大佬们点赞鼓励下吧~ 文章目录 相同的树另一颗树的子树翻…

linux死锁问题和cpu使用率过高问题排查

1、问题共同点 死锁问题和cpu使用率过高都是需要我们找出对应的问题线程。 死锁问题需要我们找出哪两个线程出现了死锁情况。 cpu使用率过高需要我们找出哪个或哪些线程占用了大量的cpu。 2、命令排查 2.1、查看机器上的Java进程 jcmd或 jps2.2、查看对应Java进程的线程级别…

五子棋理解C++思想

双人五子棋项目目录&#xff1a; class Game { public:Game();void init();bool waitPlayerPutChess(Player* player, int& oldi, int& oldj);void draw();void play();bool isOver(int playerId);public:int whoWin -1; // 谁赢了&#xff08;0&#xff1a;白棋&a…

欺诈文本分类微调(七)—— lora单卡二次调优

1. 前言 模型训练是一个不断调优的过程,这注定了我们的需要多次跑同一个训练过程。在前文欺诈文本分类微调(六):Lora单卡跑的整个训练过程中,基本可以分为几步: 数据加载数据预处理模型加载定义lora参数插入微调矩阵定义训练参数构建训练器开始训练这个流程基本是固定的…

科创中心“核”动力|趋动科技:AI算力界的领跑者

近日&#xff0c;趋动科技与深信服正式推出联合解决方案。联合解决方案将深信服EDS的高性能存储与趋动科技OrionX AI算力资源池化软件、以及GeminiAI训练平台有机结合&#xff0c;整合存力与算力资源的同时&#xff0c;帮助用户建好AI平台、管好AI资源、用好AI服务。 双方已完成…

Web3链上聚合器声呐已全球上线,开启区块链数据洞察新时代

在全球区块链技术高速发展的浪潮中&#xff0c;在创新发展理念的驱动下&#xff0c;区块链领域的工具类应用备受资本青睐。 2024年8月20日&#xff0c;由生纳&#xff08;香港&#xff09;国际集团倾力打造的一款链上应用工具——“声呐链上聚合器”&#xff0c;即“声呐链上数…

Facebook的区块链战略:如何在社交媒体中实现去中心化

随着区块链技术的发展&#xff0c;Facebook&#xff08;现Meta&#xff09;正积极探索如何将这一技术整合进其社交平台中&#xff0c;以提升用户体验和数据安全。区块链技术以去中心化、透明性和不可篡改性为特点&#xff0c;为社交媒体带来了新的可能性。本文将探讨Facebook在…

嵌入式和单片机有什么区别?

目录 &#xff08;1&#xff09;什么是嵌入式&#xff1f; &#xff08;2&#xff09;什么是单片机&#xff1f; &#xff08;3&#xff09;嵌入式和单片机的共同点 &#xff08;4&#xff09;嵌入式和单片机的区别 &#xff08;1&#xff09;什么是嵌入式&#xff1f; 关…

Linux云计算 |【第二阶段】SECURITY-DAY1

主要内容&#xff1a; 监控基础&#xff08;系统监控命令、监控软件&#xff09;、Zabbix监控服务端部署、Zabbix监控客户端部署、创建监控主机、调用监控模板、自定义key、创建模板、应用集、监控项、绑定模板&#xff1b; 一、监控概述 1&#xff09;监控的目的 ① 实时报…

【Hot100】LeetCode—114. 二叉树展开为链表

目录 1- 思路技巧——借助指针 2- 实现⭐114. 二叉树展开为链表——题解思路 3- ACM 实现 原题连接&#xff1a;114. 二叉树展开为链表 1- 思路 技巧——借助指针 思路&#xff1a;通过 ① 将左子树的右下结点的 .next ——> 拼接到当前节点的右子树上。 构造 cur 指针&a…

flutter 键盘弹出 都会重新Build

原因是调用MediaQuery.of(context)后&#xff0c;点击TextField组件时会导致调用build方法。 解决方法&#xff1a;在Scaffold组件的body嵌套Builder组件&#xff0c;然后设置一个BuildContext变量&#xff0c;将Builder组件中的context传递给BuildContext变量&#xff0c;然后…

数据结构【链试结构二叉树】

&#x1f31f;个人主页&#xff1a;落叶 目录 ​编辑 实现链式结构⼆叉树 前中后序遍历&#xff1a; 遍历规则 代码实现 前序遍历&#xff1a; 中序遍历&#xff1a; 后序遍历&#xff1a; 图解遍历&#xff1a; 函数递归栈帧图&#xff1a; 结点个数以及高度等 【⼆…

每日OJ_牛客_淘宝网店(日期模拟)

目录 牛客_淘宝网店&#xff08;日期模拟&#xff09; 解析代码 牛客_淘宝网店&#xff08;日期模拟&#xff09; 淘宝网店__牛客网 解析代码 这是一个变相的日期计算器。只不过2、3、5、7、11月算1天&#xff0c;其他7个月算2天。 既然是一个变相的日期计算器&#xff0c;那…

基于STM32开发的智能空气质量监测系统

目录 引言环境准备工作 硬件准备软件安装与配置系统设计 系统架构硬件连接代码实现 系统初始化空气质量监测与处理风扇控制与状态显示Wi-Fi通信与远程监控应用场景 家庭和办公室的空气质量监测室内空气净化器的智能控制常见问题及解决方案 常见问题解决方案结论 1. 引言 空气…

单位信息宣传考核投稿方法不对让我尝尽了苦头

自从我担任单位的信息宣传员以来,便深刻体会到“信息宣传”四个字背后的重量。每月的信息宣传考核任务就像一座大山,压在我心头。起初,我像大多数同行一样,习惯于通过电子邮件向各大媒体投稿,但这种方式让我尝尽了苦头。 记得开始尝试通过邮箱投稿时,我满怀信心地将精心准备的文…