JavaPriorityQueue使用详解与实战教程
本教程深入解析Java优先队列PriorityQueue的使用。PriorityQueue是一种基于堆数据结构的特殊队列,它能保证每次取出的是优先级最高的元素。本文将详细介绍PriorityQueue的构造函数,包括默认容量、自定义比较器以及从集合初始化的方法。同时,深入讲解offer/add、poll/remove、peek、size、contains等常用方法,并分析它们的时间复杂度。此外,本文还展示了如何通过Comparator接口自定义排序规则,以及PriorityQueue与TreeSet的区别,助你理解其底层堆结构和适用场景,掌握Java优先队列的应用技巧。
Java中的PriorityQueue是一种基于堆实现的优先队列,其核心特性是每次取出优先级最高的元素。1. 它提供了多种构造函数,包括默认容量和排序方式、指定容量、自定义比较器以及从集合初始化;2. 常用方法如offer/add插入元素、poll/remove移除元素、peek查看队首、size获取大小、contains检查是否存在,其中offer更安全,poll和remove时间复杂度为O(log n),peek和size为O(1),contains为O(n);3. 可通过实现Comparator接口自定义排序规则,并在构造时传入比较器;4. 与TreeSet不同,PriorityQueue底层是堆,只保证队首有序,而TreeSet基于红黑树,所有元素有序且支持索引,但两者均不允许null元素,适用场景各有侧重。
Java中的PriorityQueue是一种特殊的队列,它允许我们按照元素的优先级来处理它们。简单来说,你可以把它想象成一个自动排序的队列,每次取出的是优先级最高的元素。

PriorityQueue的本质是一个堆(通常是小根堆),这意味着队列中的元素总是按照某种顺序排列,保证队首元素是最小(或最大,取决于你的比较器)。
解决方案
使用PriorityQueue的关键在于理解它的构造函数和常用方法。

构造函数:
PriorityQueue()
:创建一个具有默认初始容量(11)的 PriorityQueue,并根据其元素的自然顺序对其进行排序。PriorityQueue(int initialCapacity)
:创建一个具有指定初始容量的 PriorityQueue,并根据其元素的自然顺序对其进行排序。PriorityQueue(Comparator super E> comparator)
:创建一个具有默认初始容量的 PriorityQueue,并根据指定的比较器对其元素进行排序。PriorityQueue(int initialCapacity, Comparator super E> comparator)
:创建一个具有指定初始容量的 PriorityQueue,并根据指定的比较器对其元素进行排序。PriorityQueue(Collection extends E> c)
:创建一个包含指定集合中的元素的 PriorityQueue。PriorityQueue(PriorityQueue extends E> c)
:创建一个包含指定优先级队列中的元素的 PriorityQueue。PriorityQueue(SortedSet extends E> c)
:创建一个包含指定排序集合中的元素的 PriorityQueue。
常用方法:
add(E e)
/offer(E e)
:将指定的元素插入此优先级队列。offer
通常更安全,因为它在队列满时返回false
而不是抛出异常。peek()
:检索但不移除此队列的头,如果此队列为空,则返回null
。poll()
:检索并移除此队列的头,如果此队列为空,则返回null
。remove(Object o)
:从此队列中移除指定元素的单个实例(如果存在)。contains(Object o)
:如果此队列包含指定元素,则返回true
。size()
:返回此队列中的元素数量。clear()
:从此队列中移除所有元素。iterator()
: 返回在此队列中的元素上进行迭代的迭代器。
示例代码:
import java.util.PriorityQueue; import java.util.Comparator; public class PriorityQueueExample { public static void main(String[] args) { // 使用默认自然顺序的 PriorityQueue (小根堆) PriorityQueue<Integer> pq1 = new PriorityQueue<>(); pq1.add(5); pq1.add(1); pq1.add(10); pq1.add(3); System.out.println("Default PriorityQueue:"); while (!pq1.isEmpty()) { System.out.print(pq1.poll() + " "); // 输出: 1 3 5 10 } System.out.println(); // 使用自定义 Comparator 的 PriorityQueue (大根堆) PriorityQueue<Integer> pq2 = new PriorityQueue<>(Comparator.reverseOrder()); pq2.add(5); pq2.add(1); pq2.add(10); pq2.add(3); System.out.println("Custom Comparator PriorityQueue:"); while (!pq2.isEmpty()) { System.out.print(pq2.poll() + " "); // 输出: 10 5 3 1 } System.out.println(); // 使用自定义对象的 PriorityQueue PriorityQueue<Task> taskQueue = new PriorityQueue<>(Comparator.comparingInt(Task::getPriority)); taskQueue.add(new Task("Task A", 3)); taskQueue.add(new Task("Task B", 1)); taskQueue.add(new Task("Task C", 2)); System.out.println("Task PriorityQueue:"); while (!taskQueue.isEmpty()) { System.out.println(taskQueue.poll()); } } static class Task { private String name; private int priority; public Task(String name, int priority) { this.name = name; this.priority = priority; } public String getName() { return name; } public int getPriority() { return priority; } @Override public String toString() { return "Task{" + "name='" + name + '\'' + ", priority=" + priority + '}'; } } }
PriorityQueue 的时间复杂度是多少?
offer(E e)
和add(E e)
: O(log n),其中 n 是队列中的元素数量。这是因为插入元素后需要调整堆结构。poll()
和remove(Object o)
: O(log n),其中 n 是队列中的元素数量。移除元素后也需要调整堆结构。peek()
: O(1),因为只是查看堆顶元素,不需要调整堆。size()
: O(1),直接返回队列的大小。contains(Object o)
: O(n),在最坏情况下需要遍历整个队列来查找元素。
如何自定义 PriorityQueue 的排序规则?
可以通过实现 Comparator
接口来定义自定义的排序规则。Comparator
接口允许你指定两个对象之间的比较方式。 在创建 PriorityQueue
实例时,将自定义的 Comparator
传递给构造函数。
import java.util.PriorityQueue; import java.util.Comparator; public class CustomPriorityQueue { public static void main(String[] args) { // 自定义 Comparator,按照字符串长度降序排列 Comparator<String> stringLengthComparator = (s1, s2) -> s2.length() - s1.length(); PriorityQueue<String> stringQueue = new PriorityQueue<>(stringLengthComparator); stringQueue.add("apple"); stringQueue.add("banana"); stringQueue.add("kiwi"); stringQueue.add("orange"); System.out.println("Custom String PriorityQueue:"); while (!stringQueue.isEmpty()) { System.out.println(stringQueue.poll()); // 输出: banana, orange, apple, kiwi } } }
PriorityQueue 和 TreeSet 的区别是什么?
PriorityQueue
和 TreeSet
都是用于存储有序元素的集合,但它们在实现和使用上有一些关键区别:
- 底层数据结构:
PriorityQueue
基于堆(通常是小根堆)实现,而TreeSet
基于红黑树实现。 - 排序方式:
PriorityQueue
只保证队首元素是最小(或最大)的,其他元素的顺序不确定。TreeSet
则保证所有元素都是有序的。 - 是否允许 null 元素:
PriorityQueue
不允许存储null
元素,会抛出NullPointerException
。TreeSet
也不允许存储null
元素(在大多数实现中)。 - 时间复杂度:
PriorityQueue
的offer
和poll
操作的时间复杂度为 O(log n),peek
操作为 O(1)。TreeSet
的add
、remove
和contains
操作的时间复杂度都为 O(log n)。 - 应用场景:
PriorityQueue
适用于需要频繁获取最小(或最大)元素的场景,例如任务调度、堆排序等。TreeSet
适用于需要维护有序集合的场景,例如字典、索引等。
简单来说,如果只需要获取优先级最高的元素,使用 PriorityQueue
效率更高。如果需要维护一个完全有序的集合,使用 TreeSet
更合适。
到这里,我们也就讲完了《JavaPriorityQueue使用详解与实战教程》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于时间复杂度,堆,TreeSet,comparator,PriorityQueue的知识点!

- 上一篇
- Laravel自定义命令无法注册解决方法

- 下一篇
- PHP弹幕系统开发与实时交互方法
-
- 文章 · java教程 | 44分钟前 |
- SpringBoot日志配置与管理技巧
- 154浏览 收藏
-
- 文章 · java教程 | 47分钟前 |
- Java方法如何传递数组参数
- 222浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- Java多线程竞态条件:原理与解决方法
- 457浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Redis缓存与Java集成实战教程
- 132浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Java实现Zookeeper分布式锁教程
- 270浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Java实现PDF模板填充方法详解
- 451浏览 收藏
-
- 文章 · java教程 | 4小时前 |
- Selenium关闭广告弹窗方法教程
- 116浏览 收藏
-
- 文章 · java教程 | 5小时前 |
- JavaWebSocket可靠消息重发方案详解
- 304浏览 收藏
-
- 文章 · java教程 | 5小时前 |
- Room数据库预填充数据不显示怎么办
- 439浏览 收藏
-
- 文章 · java教程 | 5小时前 |
- SpringBoot非UTF-8请求配置方法
- 501浏览 收藏
-
- 文章 · java教程 | 6小时前 |
- Log4j2JsonTemplateLayout堆栈漏洞详解
- 428浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 514次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 499次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 千音漫语
- 千音漫语,北京熠声科技倾力打造的智能声音创作助手,提供AI配音、音视频翻译、语音识别、声音克隆等强大功能,助力有声书制作、视频创作、教育培训等领域,官网:https://qianyin123.com
- 1044次使用
-
- MiniWork
- MiniWork是一款智能高效的AI工具平台,专为提升工作与学习效率而设计。整合文本处理、图像生成、营销策划及运营管理等多元AI工具,提供精准智能解决方案,让复杂工作简单高效。
- 996次使用
-
- NoCode
- NoCode (nocode.cn)是领先的无代码开发平台,通过拖放、AI对话等简单操作,助您快速创建各类应用、网站与管理系统。无需编程知识,轻松实现个人生活、商业经营、企业管理多场景需求,大幅降低开发门槛,高效低成本。
- 1029次使用
-
- 达医智影
- 达医智影,阿里巴巴达摩院医疗AI创新力作。全球率先利用平扫CT实现“一扫多筛”,仅一次CT扫描即可高效识别多种癌症、急症及慢病,为疾病早期发现提供智能、精准的AI影像早筛解决方案。
- 1044次使用
-
- 智慧芽Eureka
- 智慧芽Eureka,专为技术创新打造的AI Agent平台。深度理解专利、研发、生物医药、材料、科创等复杂场景,通过专家级AI Agent精准执行任务,智能化工作流解放70%生产力,让您专注核心创新。
- 1023次使用
-
- 提升Java功能开发效率的有力工具:微服务架构
- 2023-10-06 501浏览
-
- 掌握Java海康SDK二次开发的必备技巧
- 2023-10-01 501浏览
-
- 如何使用java实现桶排序算法
- 2023-10-03 501浏览
-
- Java开发实战经验:如何优化开发逻辑
- 2023-10-31 501浏览
-
- 如何使用Java中的Math.max()方法比较两个数的大小?
- 2023-11-18 501浏览