Java并发基础:PriorityBlockingQueue全面解析!

Java并发基础:PriorityBlockingQueue全面解析! - 程序员古德

内容概要

PriorityBlockingQueue类能高效处理优先级任务,确保高优先级任务优先执行,它内部基于优先级堆实现,保证了元素的有序性,同时,作为BlockingQueue接口的实现,它提供了线程安全的队列操作,适用于多线程环境下的任务调度与资源管理,简洁而强大的API使得开发者能轻松应对复杂的并发场景。

核心概念

PriorityBlockingQueue 实现了一个线程安全的优先级队列,在这个队列中,元素根据它们的自然排序(如果它们实现了 Comparable 接口)或者传递给队列构造器的 Comparator 进行排序。

比如有一个打印服务,在这个系统中,用户可以提交打印任务,每个任务都有一个优先级,高优先级的任务(比如紧急的文档)应该比低优先级的任务(如日常报告)更快地被处理。

在这个场景中,PriorityBlockingQueue 可以用来存储和管理待处理的打印任务,每当一个新的打印任务被提交时,它就被添加到队列中,由于 PriorityBlockingQueue 是一个优先级队列,所以高优先级的任务会自动排在队列的前面。

打印服务的工作线程可以从队列中取出任务来处理,由于队列是线程安全的,多个工作线程可以同时从队列中安全地取出任务,而且,由于队列会根据优先级对任务进行排序,所以工作线程总是先处理优先级最高的任务。

PriorityBlockingQueue 主要解决以下类似场景的问题:

  1. 并发访问:在多线程环境中,PriorityBlockingQueue 提供了安全的并发访问机制,多个线程可以同时向队列中添加或检索元素,而无需担心数据的不一致性或损坏。
  2. 优先级排序:队列中的元素根据其自然排序顺序或者传递给队列构造函数的比较器(Comparator)来排序,这使得在处理任务或消息时,可以确保首先处理优先级最高的项。
  3. 资源分配:在资源有限的情况下,PriorityBlockingQueue 可以帮助确定哪些任务或请求应该首先获得资源,通过为不同的任务设置不同的优先级,系统可以优先处理更重要的任务。
  4. 任务调度:在任务调度系统中,PriorityBlockingQueue 可用于管理待执行的任务,工作线程可以从队列中检索并执行优先级最高的任务,从而确保任务按照优先级顺序执行。
  5. 缓冲和流量控制:在高负载情况下,PriorityBlockingQueue 可以作为缓冲区来存储待处理的项目,并通过其阻塞特性来控制流量,当队列满时,尝试添加元素的线程将被阻塞,直到队列中有可用空间;同样地,当队列为空时,尝试检索元素的线程也将被阻塞,直到有元素可用。
  6. 延迟执行:虽然 PriorityBlockingQueue 本身不直接支持延迟执行,但可以通过结合使用优先级和自定义的比较器来实现类似的效果,例如,可以将任务的执行时间作为优先级的一部分,并确保在执行时间之前任务不会被检索出来。

代码案例

下面是一个简单例子,演示如何使用PriorityBlockingQueue类,这个例子中,创建了一个优先级阻塞队列,用于存储和检索Task对象,这些对象根据它们的优先级进行排序,client代码会向队列中添加任务,并从队列中检索并处理优先级最高的任务,如下:

import java.util.concurrent.PriorityBlockingQueue;  
  
// 任务类,实现了Comparable接口以便能够根据优先级进行排序  
class Task implements Comparable<Task> {  
    private int priority;  
    private String description;  
  
    public Task(int priority, String description) {  
        this.priority = priority;  
        this.description = description;  
    }  
  
    public int getPriority() {  
        return priority;  
    }  
  
    public String getDescription() {  
        return description;  
    }  
  
    // 根据优先级对任务进行排序,优先级高的任务排在前面  
    @Override  
    public int compareTo(Task other) {  
        // 注意:这里使用Integer.compare进行比较,以正确处理负数优先级  
        return Integer.compare(other.priority, this.priority); // 降序排列  
    }  
  
    @Override  
    public String toString() {  
        return "Task{" + "priority=" + priority + ", description='" + description + "'}";  
    }  
}  
  
// 客户端代码,演示如何使用PriorityBlockingQueue  
public class PriorityBlockingQueueExample {  
  
    public static void main(String[] args) throws InterruptedException {  
        // 创建一个优先级阻塞队列  
        PriorityBlockingQueue<Task> queue = new PriorityBlockingQueue<>();  
  
        // 向队列中添加任务  
        queue.put(new Task(3, "Low priority task"));  
        queue.put(new Task(5, "Medium priority task"));  
        queue.put(new Task(1, "Very low priority task"));  
        queue.put(new Task(7, "High priority task"));  
  
        // 从队列中检索并处理任务,直到队列为空  
        while (!queue.isEmpty()) {  
            // take方法会阻塞,直到队列中有元素可用  
            Task task = queue.take();  
            System.out.println("Processing task: " + task);  
        }  
    }  
}

在上面的代码中,创建了一个PriorityBlockingQueue实例,并向其中添加了四个具有不同优先级的任务,然后,使用一个循环从队列中检索并处理任务,直到队列为空,由于PriorityBlockingQueue是一个优先级队列,因此当从队列中检索任务时,优先级最高的任务总是首先被取出。

compareTo方法中,使用Integer.compare(other.priority, this.priority)来对任务进行降序排列,因此,优先级数值越高的任务将被视为优先级越高,并排在队列的前面,如果想要升序排列(即优先级数值越低的任务排在前面),可以简单地调换other.prioritythis.priority的位置。

上面代码输出如下类似内容:

Processing task: Task{priority=7, description='High priority task'}  
Processing task: Task{priority=5, description='Medium priority task'}  
Processing task: Task{priority=3, description='Low priority task'}  
Processing task: Task{priority=1, description='Very low priority task'}

核心API

PriorityBlockingQueue 实现了 BlockingQueue 接口并使用优先级堆对元素进行排序,以下是 PriorityBlockingQueue 类中一些常用方法的含义:

1、构造方法

  • PriorityBlockingQueue(): 创建一个具有默认初始容量的 PriorityBlockingQueue
  • PriorityBlockingQueue(int initialCapacity): 创建一个具有指定初始容量的 PriorityBlockingQueue
  • PriorityBlockingQueue(int initialCapacity, Comparator<? super E> comparator): 创建一个具有指定初始容量,并根据指定的比较器对元素进行排序的 PriorityBlockingQueue

2、插入方法

  • add(E e): 将指定的元素插入到此队列中,如果成功,则返回 true(由于队列没有容量限制,因此该方法总是返回 true,除非元素为 null)。
  • offer(E e): 将指定的元素插入到此队列中,并立即返回。该方法等效于 add(E e)
  • put(E e) throws InterruptedException: 将指定的元素插入到此队列中,等待必要的空间变得可用,如果当前线程被中断,则抛出 InterruptedException

3、移除方法

  • remove(Object o): 移除队列中首次出现的指定元素(如果存在)。
  • poll(): 检索并移除此队列的头,如果此队列为空,则返回 null
  • take() throws InterruptedException: 检索并移除此队列的头,等待元素变得可用,如果当前线程被中断,则抛出 InterruptedException

4、检查方法

  • peek(): 检索,但不移除此队列的头,如果此队列为空,则返回 null
  • element() throws NoSuchElementException: 检索,但不移除此队列的头,如果此队列为空,则抛出 NoSuchElementException

5、其他方法

  • size(): 返回队列中的元素数量。
  • isEmpty(): 如果队列为空,则返回 true
  • clear(): 移除此队列中的所有元素。
  • contains(Object o): 如果队列包含指定的元素,则返回 true
  • remainingCapacity(): 由于 PriorityBlockingQueue 没有容量限制,此方法始终返回 Integer.MAX_VALUE
  • drainTo(Collection<? super E> c): 移除此队列中所有可用的元素,并将它们添加到给定的集合中。
  • drainTo(Collection<? super E> c, int maxElements): 移除此队列中最多给定数量的可用元素,并将它们添加到给定的集合中。
  • toArray(): 返回包含队列中所有元素的数组。
  • iterator(): 返回在此队列元素上进行迭代的迭代器。注意,由于并发修改的可能性,迭代器的行为是弱一致的。
  • comparator(): 返回用于对此队列中的元素进行排序的比较器,如果此队列按其自然顺序排序,则返回 null

核心总结

Java并发基础:PriorityBlockingQueue全面解析! - 程序员古德

PriorityBlockingQueue类允许开发者存储元素并根据其自然排序或者提供的Comparator进行排序,其优点在于它能高效地处理需要优先级调度的任务,确保最高优先级的任务总是优先被处理,它的缺点是在高并发场景下,由于线程间的竞争,性能可能会受到影响,此外,虽然它提供了并发安全性,但在迭代过程中并不保证元素的顺序一致性。

关注我,每天学习互联网编程技术 - 程序员古德

END!

往期回顾

Java并发基础:DelayQueue全面解析!

Java并发基础:LinkedBlockingDeque全面解析!

Java并发基础:LinkedTransferQueue全面解析!

Java并发基础:LinkedBlockingQueue全面解析!

Java并发基础:Deque接口和Queue接口的区别?

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

系统架构26 - 软件架构设计(5)

特定领域软件体系结构 定义不同定义必备特征领域 基本活动领域分析领域设计领域实现 参与人员建立过程 特定领域软件体系结构的主要目的是在一组相关的应用中共享软件体系结构。 定义 DSSA (Domain Specific Software Architecture) 就是在一个特定应用领域中为一组应用提供组…

算法-16-并查集

并查集简介 并查集&#xff1a;一开始&#xff0c;把a&#xff0c;b&#xff0c;c放入并查集&#xff0c;a自己一个集合&#xff0c;b自己一个&#xff0c;c自己一个 提供的方法 1.boolean isSameSet(a,b)&#xff0c;判断ab是否在同一个集合 2.void union(a,b),把a所…

基于PHP的学生管理系统

前言 基于PHP的学生管理系统&#xff1b; 实现 登录、注册、学生信息、修改学生、删除学生、查询学生、添加学生等功能 &#xff1b; 环境准备 开发平台&#xff1a;PhpStrom2022.1.2 、Phpstudy_pro 数据库&#xff1a;MySQL5.7.26 技术架构 Bootstrap PHP7.3.4html5css3 项目…

vue安装使用less,解决与webpack的冲突

第077个 查看专栏目录: VUE ------ element UI 专栏目标 在vue和element UI联合技术栈的操控下&#xff0c;本专栏提供行之有效的源代码示例和信息点介绍&#xff0c;做到灵活运用。 提供vue2的一些基本操作&#xff1a;安装、引用&#xff0c;模板使用&#xff0c;computed&a…

C++-带你深度理解string类的常见接口

1. 为什么学习string类&#xff1f; C语言中&#xff0c;字符串是以\0结尾的一些字符的集合&#xff0c;为了操作方便&#xff0c;C标准库中提供了一些str系列的库函数&#xff0c;但是这些库函数与字符串是分离开的&#xff0c;不太符合OOP的思想&#xff0c;而且底层空间需…

LeetCode周赛——384

1.修改矩阵&#xff08;模拟&#xff09; class Solution { public:vector<vector<int>> modifiedMatrix(vector<vector<int>>& matrix) {int n matrix.size();int m matrix[0].size();vector<int> ans(m);for(int i 0; i < m; i)for(…

如何写好一个简历

如何编写求职简历 论Java程序员求职中简历的重要性 好简历的作用 在求职过程中&#xff0c;一份好的简历是非常重要的&#xff0c;它甚至可以直接决定能否被面试官认可。一份出色或者说是成功的个人简历&#xff0c;最根本的作用是能让看这份简历的人产生一定要见你的强烈愿…

腾讯云4核8G服务器够用吗?来看看支持多少人访问!

腾讯云4核8G服务器支持多少人在线访问&#xff1f;支持25人同时访问。实际上程序效率不同支持人数在线人数不同&#xff0c;公网带宽也是影响4核8G服务器并发数的一大因素&#xff0c;假设公网带宽太小&#xff0c;流量直接卡在入口&#xff0c;4核8G配置的CPU内存也会造成计算…

基于 Python 的大数据的电信反诈骗系统

博主介绍&#xff1a;✌程序员徐师兄、7年大厂程序员经历。全网粉丝12w、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专栏推荐订阅&#x1f447;…

【c语言】字符串常见函数 下

&#x1f388;个人主页&#xff1a;甜美的江 &#x1f389;欢迎 &#x1f44d;点赞✍评论⭐收藏 &#x1f917;收录专栏&#xff1a;c语言 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共同学习、交流进步&a…

文生图提示词:视角选择

构图和视角 --视角选择 Perspective Choices 涵盖从基本的摄影视角到更复杂和专业的视角技巧&#xff0c;展示了在视觉艺术创作中可以采用的多样化视角选择。 Eye Level 眼平线视角 High Angle 高角度 Low Angle 低角度 Birds Eye View 鸟瞰视角 Worms Eye View 虫视视角 Front…

【51单片机】直流电机驱动(PWM)(江科大)

1.直流电机介绍 直流电机是一种将电能转换为机械能的装置。一般的直流电机有两个电极,当电极正接时,电机正转,当电极反接时,电机反转 直流电机主要由永磁体(定子)、线圈(转子)和换向器组成 除直流电机外,常见的电机还有步进电机、舵机、无刷电机、空心杯电机等 2.电机驱动…

使用UMAP降维可视化RAG嵌入

大型语言模型&#xff08;LLMs&#xff09;如 GPT-4 已经展示了出色的文本理解和生成能力。但它们在处理领域特定信息方面面临挑战&#xff0c;比如当查询超出训练数据范围时&#xff0c;它们会产生错误的答案。LLMs 的推理过程也缺乏透明度&#xff0c;使用户难以理解达成结论…

Qt之条件变量QWaitCondition详解

QWaitCondition内部实现结构图&#xff1a; 相关系列文章 C之Pimpl惯用法 目录 1.简介 2.示例 2.1.全局配置 2.2.生产者Producer 2.3.消费者Consumer 2.4.测试例子 3.原理分析 3.1.辅助函数CreateEvent 3.2.辅助函数WaitForSingleObject 3.3.QWaitConditionEvent …

C++笔记1:操纵符输入输出

C操纵符用来控制输出控制&#xff0c;一是输出的形式&#xff0c;二是控制补白的数量和位置。本文记录一下&#xff0c;在一些笔试的ACM模式可能有用。其中1-4节的部分是关于格式化输入输出操作&#xff0c;5-6节的部分是关于未格式化输入输出操作。 1. 控制布尔值的格式 一般…

uniapp API文档地址 以及 HBuilder安装

uniapp API文档地址 以及 HBuilder安装 一、进入 当前网站 uni-app 官网 [uni-app](https://zh.uniapp.dcloud.io/quickstart-hx.html)二、点击截图下载文件 三、 进入 当前网站 &#xff08;https://www.dcloud.io/hbuilderx.html&#xff09; 浏览器会识别 也可以自行选择…

基于GIS、RS、VORS模型、CCDM模型、geodetecto、GWR模型集成的生态系统健康的耦合协调分析

详情点击公众号&#xff1a;技术科研吧 链接&#xff1a;基于GIS、RS、VORS模型、CCDM模型、geodetecto、GWR模型集成的生态系统健康的耦合协调分析 前沿 当空间大数据、云计算与人工智能发生碰撞,地理服务产业也不断发生变革与进步。ArcGIS Pro 是一个专业的桌面 GIS 应用程…

17 外排序

排序分为内排序和外排序&#xff0c;内排序是在内存中的排序。外排序指在磁盘中文件的排序&#xff0c;因为在磁盘中&#xff0c;不能进行下标访问&#xff0c;归并排序经常用于磁盘中文件的排序 假如有10亿个整形数据在磁盘中&#xff0c;要对它排序&#xff0c;内存中只有1G…

水果FL Studio21.2最新中文版功能特点介绍

FL Studio 21的特点和优势包括&#xff1a; 丰富的主题换肤&#xff1a;用户可以通过调整色调、饱和度、亮度、文本、仪表和步进序列器的颜色&#xff0c;来个性化定制FL Studio 21的外观&#xff0c;使其更符合个人审美或工作风格。更快的音频编辑&#xff1a;FL Studio 21集…

AI少女/HS2甜心选择2 仿剑三剑灵人物卡全合集打包

AI少女/HS2甜心选择2 仿剑三剑灵人物卡全合集打包 内含&#xff1a;菩提禅音[剑网3]明教晓天喵姐[剑3]明教晓天喵姐无帽版[剑3]茱莉亚[剑灵] 下载地址&#xff1a; https://www.changyouzuhao.cn/12492.html