Java 入门指南:Java 并发编程 —— 并发容器 PriorityBlockingQueue

BlockingQueue

BlockingQueue 是Java并发包(java.util.concurrent)中提供的一个阻塞队列接口,它继承自 Queue 接口。

BlockingQueue 中的元素采用 FIFO 的原则,支持多线程环境并发访问,提供了阻塞读取和写入的操作,当前线程在队列满或空的情况下会被阻塞,直到被唤醒或超时。

常用的实现类有:

  • ArrayBlockingQueue:并发容器 ArrayBlockingQueue 详解
  • LinkedBlockingQueue:并发容器 LinkedBlockingQueue 详解
  • PriorityBlockingQueue
  • SynchronousQueue
  • LinkedBlockingDeque:并发容器 LinkedBlockingDeque 详解
    等类,它们的实现方式各有不同。
适用场景

BlockingQueue 通常用于一个线程生产对象,而另外一个线程消费这些对象的场景。

![[BlockingQueue Model.png]]

一个线程将会持续生产新对象并将其插入到队列之中,直到队列达到它所能容纳的临界点

如果该阻塞队列到达了其临界点,生产者线程将会在往里边插入新对象时发生阻塞。它会一直处于阻塞之中,直到消费者线程从队列中拿走一个对象。

消费者线程将会一直从该阻塞队列中拿出对象。如果消费线程尝试去从一个空的队列中提取对象的话,这个消费线程将会处于阻塞之中,直到一个生产线程把一个对象丢进队列。

常用方法
  1. put(E e):将元素 e 插入到队列中,如果队列已满,则会阻塞当前线程,直到队列有空闲空间

  2. offer(E e):将元素 e 插入到队列中,如果队列已满,则返回 false。

  3. offer(E element, long timeout, TimeUnit unit) 方法是 BlockingQueue:在指定的时间内将元素添加到队列中。

    • timeout:超时时间,表示在指定的时间内等待队列空间可用。如果超过指定的时间仍然无法将元素添加到队列中,将返回 false。

    • unit:超时时间的单位。

  4. take():移除并返回队列头部的元素,如果队列为空,则会阻塞当前线程,直到队列有元素

  5. poll():移除并返回队列头部的元素,如果队列为空,则返回 null

  6. poll(long timeout, TimeUnit unit):在指定的时间内从队列中检索并移除元素。返回移除的元素。如果超过指定的时间仍然没有可用的元素,将返回 null。

  7. peek():返回队列头部的元素,但不会移除。如果队列为空,则返回null

  8. size():返回队列中元素的数量

  9. isEmpty():判断队列是否为空,为空返回 true,否则返回 false

  10. isFull():判断队列是否已满,已满返回 true,否则返回 false

  11. clear():清空队列中的所有元素

行为抛异常返回特定值阻塞超时
插入add(o)offer(o)put(o)offer(o, timeout, timeunit)
移除remove()poll()take()poll(timeout, timeunit)
检查element()peek()
  • 抛异常: 如果试图的操作无法立即执行,抛一个异常。

  • 特定值: 如果试图的操作无法立即执行,返回 true / false / null)。

  • 阻塞: 如果试图的操作无法立即执行,该方法调用将会发生阻塞,直到能够执行。

  • 超时: 如果试图的操作无法立即执行,该方法调用将会发生阻塞,直到能够执行,但等待时间不会超过给定值。返回一个特定值以告知该操作是否成功(典型的是 true / false)。

若向 BlockingQueue 中插入 null,将会抛出 NullPointerException

死锁问题

需要注意的是,在使用 BlockingQueue 时要注意防止死锁的问题:

  • 在队列满之后调用 offer() 方法插入元素会返回false,此时不能直接调用put()方法,因为在插入之前还需要获取其它资源,如果在获取资源时一直阻塞在这里,就会发生死锁。

  • 为了防止死锁的问题,建议使用 offer(E e, long timeout, TimeUnit unit)poll(long timeout, TimeUnit unit) 带有超时时间的方法。

PriorityBlockingQueue

PriorityBlockingQueue 是一个无界的并发队列。它使用了和类PriorityQueue 一样的排序规则。

所有插入到 PriorityBlockingQueue 的元素必须实现 java.lang.Comparable 接口。因此该队列中元素的排序就取决于自定义的 Comparable 实现。如果元素没有实现 Comparable 接口并且没有提供自定义的比较器,将会引发 ClassCastException

PriorityBlockingQueue 对于具有相等优先级(compare() == 0)的元素并不强制任何特定行为。

特点
  • 线程安全PriorityBlockingQueue是线程安全的,它使用锁(如ReentrantLock)来确保在多线程环境下的数据一致性和操作的原子性。

  • 优先级排序:队列中的元素会根据它们的优先级进行排序,默认情况下,元素按照其自然顺序(如果实现了Comparable接口)进行排序,或者根据构造时提供的Comparator进行排序。

  • 阻塞特性:当队列为空时,尝试从队列中取出元素的线程会被阻塞,直到队列中有元素可用;当队列已满(对于无界队列,实际上是系统资源耗尽)时,尝试向队列中添加元素的线程也会被阻塞,或者可以选择非阻塞操作(如offer方法)。

  • 无界扩容PriorityBlockingQueue 是基于数组实现的,当队列中的元素数量超过当前数组的容量时,它会自动进行扩容操作,以确保能够继续存储新的元素。

扩容问题

PriorityBlockingQueue 并不是一个动态调整容量的队列。它的“无界”特性并不是指它会动态地调整其内部存储的大小,而是指它不会因为容量限制而阻止新的元素被加入队列。相反,它会尽可能多地存储元素,直到耗尽系统的可用内存。

尽管 PriorityBlockingQueue 没有显式的容量限制,但在实际应用中,仍然需要考虑以下几点:

  1. 内存限制:虽然 PriorityBlockingQueue 没有固定的容量限制,但它仍然受到 JVM 的堆内存限制。如果队列中的元素过多,会导致内存不足,最终可能导致 OutOfMemoryError

  2. 垃圾回收:当队列中的元素被消费后,如果没有其他引用指向这些元素,垃圾回收器会自动回收这些对象所占用的内存。因此,即使队列曾经存储了大量的元素,只要这些元素被消费并且没有其他引用,内存就会被释放。

应用场景

PriorityBlockingQueue 适用于多种场景,包括但不限于:

  • 任务调度:在任务调度系统中,可以使用 PriorityBlockingQueue 来管理待执行的任务,工作线程可以从中取出优先级最高的任务来执行。

  • 资源管理:在资源有限的情况下,可以使用 PriorityBlockingQueue 来确定哪些请求或任务应该优先获得资源。

  • 并发访问:在多线程环境中,PriorityBlockingQueue 提供了安全的并发访问机制,多个线程可以同时向队列中添加或检索元素。

构造方法
  1. 创建一个初始容量为 11(默认容量)PriorityBlockingQueue 对象。
PriorityBlockingQueue()
  1. 创建一个初始容量为指定容量的 PriorityBlockingQueue 对象。
PriorityBlockingQueue(int capacity)
  1. 创建一个初始元素集合为指定集合的 PriorityBlockingQueue 对象。
PriorityBlockingQueue(Collection<? extends E> collection)
  1. 创建一个初始容量为指定容量、元素按照指定比较器优先级排序的 PriorityBlockingQueue 对象。
PriorityBlockingQueue(int c, Comparator<? super E> comparator)
使用示例

下面是一个使用 PriorityBlockingQueue 的简单示例,展示如何实现一个基于优先级的任务队列:

import java.util.concurrent.PriorityBlockingQueue;
import java.util.concurrent.TimeUnit;class Task implements Comparable<Task> {private final int priority;private final String description;public Task(int priority, String description) {this.priority = priority;this.description = description;}public int getPriority() {return priority;}public String getDescription() {return description;}@Overridepublic int compareTo(Task other) {// 优先级数值越小,优先级越高return Integer.compare(this.priority, other.priority);}@Overridepublic String toString() {return "Task{" +"priority=" + priority +", description='" + description + '\'' +'}';}
}public class PriorityBlockingQueueExample {public static void main(String[] args) {// 创建一个 PriorityBlockingQueue 实例PriorityBlockingQueue<Task> taskQueue = new PriorityBlockingQueue<>();// 创建生产者线程Thread producer = new Thread(() -> {try {// 生成不同优先级的任务taskQueue.put(new Task(3, "Medium Priority Task"));taskQueue.put(new Task(1, "High Priority Task"));taskQueue.put(new Task(5, "Low Priority Task"));taskQueue.put(new Task(2, "Normal Priority Task"));System.out.println("所有任务已添加到队列中");} catch (InterruptedException e) {Thread.currentThread().interrupt(); // 设置线程中断状态System.err.println("生产者线程被中断");}});// 创建消费者线程Thread consumer = new Thread(() -> {while (true) {try {Task task = taskQueue.take();System.out.println("处理任务: " + task);TimeUnit.MILLISECONDS.sleep(1000); // 模拟任务处理时间} catch (InterruptedException e) {Thread.currentThread().interrupt(); // 设置线程中断状态System.err.println("消费者线程被中断");break; // 终止循环}}});// 启动生产者线程和消费者线程producer.start();consumer.start();// 等待生产者线程结束try {producer.join();} catch (InterruptedException e) {Thread.currentThread().interrupt(); // 设置线程中断状态System.err.println("主线程被中断");}// 中断消费者线程consumer.interrupt();}
}

示例说明:

  • 优先级:在这个例子中,优先级数值越小,优先级越高。因此,优先级为 1 的任务会被首先处理。
  • 队列无界PriorityBlockingQueue 默认是无界的,即没有固定的容量限制。但是,如果队列中的元素数量过大,可能会导致内存溢出。
  • 线程中断:在捕获 InterruptedException 异常时,需要调用 Thread.currentThread().interrupt() 来恢复线程的中断状态,以便其他可能依赖中断状态的部分能够正确响应。
  • 消费者线程的终止:在生产者线程结束后,通过中断消费者线程来终止其无限循环。这是为了防止消费者线程无限期地运行下去。

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

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

相关文章

视频汇聚平台LntonAIServer视频质量诊断功能--偏色检测与噪声检测

随着视频监控技术的不断进步&#xff0c;视频质量成为了决定监控系统性能的关键因素之一。LntonAIServer新增的视频质量诊断功能&#xff0c;特别是偏色检测和噪声检测&#xff0c;进一步强化了视频监控系统的可靠性和实用性。下面我们将详细介绍这两项功能的技术细节、应用场景…

【AI】Pytorch_损失函数优化器

建议点赞收藏关注&#xff01;持续更新至pytorch大部分内容更完。 本文已达到10w字&#xff0c;故按模块拆开&#xff0c;详见目录导航。 整体框架如下 数据及预处理 模型及其构建 损失函数及优化器 本节目录 损失函数创建损失函数 &#xff08;共18个&#xff09;nn.CrossEnt…

【多场景应用】基于杰发科技AC7840x的Mini LED背光驱动设计

应用场景&#xff1a; 在汽车应用中&#xff0c;Mini LED背光驱动设计主要用于仪表盘、中控屏和车载娱乐系统等显示屏。这项技术可以显著提升显示效果&#xff0c;提供更高的亮度、更深的黑色和更广的色域&#xff0c;使得图像更加生动逼真&#xff0c;尤其在强光和宽温度范围…

Redis 篇-深入了解查询缓存与缓存所带来的问题(读写不一致、缓存穿透、缓存雪崩、缓存击穿)

&#x1f525;博客主页&#xff1a; 【小扳_-CSDN博客】 ❤感谢大家点赞&#x1f44d;收藏⭐评论✍ 本章目录 1.0 什么是缓存 2.0 项目中具体如何添加缓存 3.0 添加缓存后所带来的问题 3.1 读写不一致问题 3.1.1 缓存更新策略 3.1.2 具体实现缓存与数据库的双写一致 3.2 缓存穿…

【日记】想见珍一面怎么就这么难(985 字)

正文 想见珍一面怎么就这么难…… 事故频发。昨天说考试时间跟机票时间冲突了&#xff0c;最后结果出来了&#xff0c;改签了&#xff0c;并且差价不补。我不干&#xff0c;他们也不干。因为上级行给我们行长施压&#xff0c;于是我们行长给我施压。最后要到了国庆之前拔智齿的…

华为 HCIP-Datacom H12-821 题库 (6)

有需要题库的可以看主页置顶 V群仅进行学习交流 1.转发表中 FLAG 字段中B 的含义是&#xff1f; A、可用路由 B、静态路由 C、黑洞路由 D、网关路由 答案&#xff1a;C 解析&#xff1a; 可用路由用U 表示&#xff0c;静态路由用 S 表示&#xff0c;黑洞路由用 B 表示&#x…

笔试,牛客.kotori和n皇后​,牛客.AOE还是单体

目录 牛客.kotori和n皇后​编辑 牛客.AOE还是单体 牛客.kotori和n皇后 想起来&#xff0c;我之前还写过n皇后的题&#xff0c;但是这个我开始只能想到暴力解法 判断是不是斜对角线&#xff0c;联想yxb和y-xb,假如在一条线上&#xff0c;那么他们的x和y会对应成比例&#xff0c…

【弱监督时间动作定位】Probabilistic Vision-Language Representation for WSTAL 论文阅读

Probabilistic Vision-Language Representation for Weakly Supervised Temporal Action Localization 论文阅读 Abstract1 Introduction2 RELATEDWORK2.1 Weakly Supervised Temporal Action Localization2.2 Vision Language Pre-training2.3 Probabilistic Representation 3…

RocketMQ高级特性四-消息过滤

目录 前言 Broker端过滤 定义与概述 消息过滤分类 原理机制 使用场景 优缺点 Java代码示例 - Tag过滤 Java代码示例 - SQL92过滤 客户端过滤 定义与概述 原理机制 使用场景 优缺点 Java代码示例 总结 前言 消息过滤是RocketMQ的一项高级特性&#xff0c;它允许…

常见HTTP状态码、APUD响应状态字及含义

目录 一、HTTP状态码 二、APDU指令码 一、HTTP状态码 HTTP状态&#xff08;HTTP Status Code&#xff09;是用以表示网页服务器超文本传输协议响应状态的3位数字代码。 关于HTTP状态码更加详细介绍推荐阅读&#xff1a; http://t.csdnimg.cn/qSJv6http://t.csdnimg.cn/qSJv…

光敏电阻传感器详解(STM32)

目录 一、介绍 二、传感器原理 1.光敏电阻传感器介绍 2.原理图 三、程序设计 main.c文件 ldr.h文件 ldr.c文件 四、实验效果 五、资料获取 项目分享 一、介绍 光敏电阻器是利用半导体的光电导效应制成的一种电阻值随入射光的强弱而改变的电阻器&#xff0c;又称为光…

基于树莓派的儿童音频播发器—Yoto

Raspberry Pi 的开发可能性使吸引人的、以儿童为中心的音频播放器得以成型 Yoto Player 为孩子们提供了拥有和控制的绝佳体验&#xff0c;同时不会增加屏幕时间。得益于 Raspberry Pi 以及我们认可的经销商提供的支持和专业知识&#xff0c;Yoto Player 在英国取得了成功。 Yo…

七款最佳的渗透测试工具(非常详细)零基础入门到精通,收藏这一篇就够了

渗透测试工具是模拟对计算机系统、网络或 Web 应用程序的网络攻击的软件应用程序&#xff0c;它们的作用是在实际攻击者之前发现安全漏洞。它们可以作为系统的压力测试&#xff0c;揭示哪些区域可能会受到真正的威胁。 本文我将介绍七款最佳的渗透测试工具。 1 Kali Linux K…

Maven入门:自动化构建工具的基本概念与配置

一、什么是Maven 目前无论使用IDEA还是Eclipse等其他IDE&#xff0c;使用里面 ANT 工具帮助我们进行编译&#xff0c;打包运行等工作。Apache基于ANT进行了升级&#xff0c;研发出了全新的自动化构建工具Maven。 Maven使用项目对象模型&#xff08;POM-Project Object Model&…

视频合并在线工具哪个好?好用的视频合并工具推荐

当我们手握一堆零散却各有千秋的视频片段时&#xff0c;是否曾幻想过它们能像魔法般合并成一部完整、流畅的故事&#xff1f; 别担心&#xff0c;今天咱们就来一场“视频合并大冒险”&#xff0c;揭秘几款视频合并软件手机免费工具&#xff0c;帮助你在指尖上实现创意无限的视…

四、配置三层交换实验组网

一、实验拓扑 二、实验目的 通过配置交换机&#xff0c;令不同vlan间的主机能够互相通信 三、实验步骤 SW12 <Huawei>undo terminal monitor Info: Current terminal monitor is off. <Huawei>system-view Enter system view, return user view with CtrlZ. [H…

EDIUS X 10.34.9631 视频剪辑软件 下载 包含安装说明

下载地址(资源制作整理不易&#xff0c;下载使用需付费&#xff0c;不能接受请勿浪费时间下载) 链接&#xff1a;https://pan.baidu.com/s/1P2wKxVcSx5WzAtHXCaAp5A?pwd227i 提取码&#xff1a;227i

【Linux网络】应用层协议HTTP(1)

&#x1f389;博主首页&#xff1a; 有趣的中国人 &#x1f389;专栏首页&#xff1a; Linux网络 &#x1f389;其它专栏&#xff1a; C初阶 | C进阶 | 初阶数据结构 小伙伴们大家好&#xff0c;本片文章将会讲解 应用层协议HTTP 的相关内容。 如果看到最后您觉得这篇文章写得…

「深入理解」HTML Meta标签:网页元信息的配置

「深入理解」HTML Meta标签&#xff1a;网页元信息的配置 HTML的<meta>元素用于提供关于HTML文档的元数据&#xff08;metadata&#xff09;&#xff0c;这些信息对于浏览器和其他处理HTML文档的应用程序来说是非常有用的&#xff0c;如&#xff1a;<base>、<li…

【网络安全】服务基础第一阶段——第九节:Windows系统管理基础---- Windows_AD域

目录 一、域与活动目录 1.1 工作组 1.2 域 1.2.1 域&#xff08;Domain&#xff09; 1.2.2 域控制器&#xff08;Domain Controller&#xff0c;DC&#xff09; 1.2.3 功能和角色 1.2.4 管理和监控 1.2 5 域结构 1.3 组织单元&#xff08;Organizational Unit&#xff…