0

0

Java中PriorityQueue的基础使用方法

P粉602998670

P粉602998670

发布时间:2025-09-22 18:00:02

|

795人浏览过

|

来源于php中文网

原创

PriorityQueue是Java中基于优先级出队的队列,默认为小顶堆,每次取出最小元素;其核心方法包括add/offer入队、poll出队、peek查看队首;与普通FIFO队列不同,它按元素优先级排序而非入队顺序;可通过实现Comparable接口或传入Comparator自定义排序规则;常用于Dijkstra算法、任务调度、Top K问题等需动态处理最高优先级元素的场景。

java中priorityqueue的基础使用方法

Java里的

PriorityQueue
,说白了,就是一种特殊的队列。它不像我们常说的先进先出(FIFO)队列那样,而是根据元素的优先级来决定谁先出队。默认情况下,它是一个小顶堆,这意味着每次你取元素,拿到的总是当前队列里最小的那个。这在很多需要“总是处理最紧急或最小/最大”元素的场景下,简直是神来之笔。

我个人觉得,理解

PriorityQueue
的基础用法并不复杂,核心就是那么几个方法。

首先,创建它。如果你想用默认的自然排序(比如数字从小到大),直接

new PriorityQueue()
就行。如果你处理的是自定义对象,或者想实现大顶堆(从大到小),那就得传入一个
Comparator
。比如:

// 默认小顶堆,存储整数
PriorityQueue minHeap = new PriorityQueue<>();

// 大顶堆,存储整数
PriorityQueue maxHeap = new PriorityQueue<>((a, b) -> b - a); // 或者 Comparator.reverseOrder()

// 存储自定义对象,假设有一个Task类,根据priority字段排序
class Task {
    String name;
    int priority; // 优先级,数字越小优先级越高

    public Task(String name, int priority) {
        this.name = name;
        this.priority = priority;
    }

    @Override
    public String toString() {
        return "Task{" + "name='" + name + '\'' + ", priority=" + priority + '}';
    }
}

PriorityQueue taskQueue = new PriorityQueue<>(Comparator.comparingInt(t -> t.priority));

接着是往里面加元素,用

add()
或者
offer()
都可以。它们俩在功能上几乎没区别
offer()
在容量受限的队列中可能返回
false
,但
PriorityQueue
是无界的,所以一般都成功。

立即学习Java免费学习笔记(深入)”;

minHeap.add(3);
minHeap.offer(1);
minHeap.add(2);
System.out.println("当前队列 (内部表示,不保证顺序): " + minHeap); // 内部顺序不保证,但poll()会取1

然后就是取元素。

peek()
是查看队首元素,但不移除;
poll()
是取出队首元素并移除。这两个方法是用的最多的。

System.out.println("队首元素 (不移除): " + minHeap.peek()); // 输出 1
System.out.println("移除队首元素: " + minHeap.poll());    // 输出 1
System.out.println("移除后的队首元素: " + minHeap.peek()); // 输出 2
System.out.println("队列大小: " + minHeap.size());        // 输出 2
System.out.println("队列是否为空: " + minHeap.isEmpty()); // 输出 false

有时候,我发现新手会忘记

poll()
会移除元素,导致后续操作出错,所以在使用时务必明确是想“看一眼”还是“拿走”。

PriorityQueue与普通队列的本质区别在哪里?

说实话,这个问题我被问过好多次,也自己琢磨过。最核心的区别,我觉得,就在于“出队顺序”的决定机制。普通队列,比如

LinkedList
实现的
Queue
,是严格的先进先出(FIFO),你第一个放进去的,就第一个出来。这就像排队买票,谁先来谁先买。

PriorityQueue
就不同了,它关注的是“优先级”。谁的优先级高,谁就先出来,跟进队顺序无关。这就像医院的急诊室,病人来了不按先来后到,而是根据病情轻重(优先级)来决定谁先看医生。这种内在的排序机制,使得
PriorityQueue
在处理那些需要动态调整处理顺序的场景时,显得异常强大。它的底层通常是用一个二叉堆(min-heap或max-heap)来实现的,这保证了
poll()
add()
操作的平均时间复杂度是O(log n),效率相当不错。所以,如果你需要一个总是能拿到“最紧急”或“最小/最大”元素的集合,
PriorityQueue
就是你的不二之选。

如何灵活地自定义PriorityQueue的排序规则?

自定义排序规则是

PriorityQueue
非常实用的一个特性,也是我个人觉得它魅力所在的地方。Java提供了两种主要方式来实现这一点。

简单实用的Bootstrap选项卡效果
简单实用的Bootstrap选项卡效果

这是一款基于Bootstrap的简单实用的选项卡效果。该选项卡在原生boostrap选项卡的基础上进行了一些美化,效果时尚大方,非常不错。 使用方法 在页面中引入jquery和bootstrap相关文件。

下载

第一种是让你的元素类实现

Comparable
接口。如果你的对象有“自然顺序”,比如数字的大小、字符串的字典序,那么让它实现
Comparable
,并重写
compareTo
方法,
PriorityQueue
在创建时就会自动使用这个自然顺序。

class Student implements Comparable {
    String name;
    int score;

    public Student(String name, int score) {
        this.name = name;
        this.score = score;
    }

    // 假设分数越高优先级越高,所以是降序
    @Override
    public int compareTo(Student other) {
        return other.score - this.score; // 大顶堆,分数高的先出
    }

    @Override
    public String toString() {
        return "Student{" + "name='" + name + '\'' + ", score=" + score + '}';
    }
}

// 使用自然排序(即Student类中定义的compareTo)
PriorityQueue highScorers = new PriorityQueue<>();
highScorers.add(new Student("Alice", 90));
highScorers.add(new Student("Bob", 95));
highScorers.add(new Student("Charlie", 88));
System.out.println("最高分学生: " + highScorers.poll()); // Bob

第二种,也是更灵活的方式,是给

PriorityQueue
的构造函数传入一个
Comparator
对象。这种方式的好处是,你可以在不修改元素类的情况下,为同一个类定义多种排序规则。这在处理第三方库的类或者需要动态改变排序逻辑时特别有用。

// 使用Lambda表达式定义Comparator,实现按名字长度排序(短的优先)
PriorityQueue byLength = new PriorityQueue<>(Comparator.comparingInt(String::length));
byLength.add("apple");
byLength.add("banana");
byLength.add("cat");
System.out.println("按长度排序: " + byLength.poll()); // cat

// 或者,如果想实现大顶堆(最大值优先),可以这样
PriorityQueue customMaxHeap = new PriorityQueue<>((a, b) -> b - a);
customMaxHeap.add(10);
customMaxHeap.add(20);
customMaxHeap.add(5);
System.out.println("自定义大顶堆: " + customMaxHeap.poll()); // 20

我个人偏爱使用

Comparator
,特别是Java 8引入Lambda表达式后,写起来简直不要太方便,代码也更简洁明了。它把排序逻辑和数据结构本身的创建清晰地分开了,这在维护大型项目时尤其重要。

PriorityQueue在哪些实际场景中能发挥巨大作用?

说起

PriorityQueue
的应用场景,那可真是太广了,很多算法和系统设计都离不开它。在我看来,它就是那种“看着不起眼,但一用起来就离不开”的数据结构。

一个经典的例子是Dijkstra最短路径算法。在寻找图中两点间最短路径时,我们总是需要优先处理那些当前已知路径最短的节点。

PriorityQueue
就能完美地存储这些待处理的节点,并根据它们的当前最短距离进行排序,每次取出距离最小的节点进行扩展。没有它,Dijkstra算法的效率会大打折扣。

再比如,任务调度系统。想象一下,你有一堆任务,每个任务都有一个优先级或者一个计划执行时间。你需要一个调度器,总是能把当前最紧急或者最先到期的任务拿出来执行。

PriorityQueue
在这里就能大显身手,它能确保你总是能高效地获取到下一个需要执行的任务。

还有,“Top K”问题。比如从海量数据中找出最大的K个元素,或者最小的K个元素。这时,我们可以维护一个大小为K的

PriorityQueue
。如果是找最大的K个,就用一个小顶堆,遍历数据时,如果当前元素比堆顶元素大,就替换掉堆顶。如果是找最小的K个,就用一个大顶堆,逻辑类似。这种方法比直接排序整个数据集要高效得多,尤其是在数据量巨大的时候。

// 找出数组中最大的K个元素 (使用小顶堆)
int[] nums = {3, 2, 1, 5, 6, 4};
int k = 3;
PriorityQueue topKHeap = new PriorityQueue<>(); // 默认小顶堆

for (int num : nums) {
    topKHeap.offer(num);
    if (topKHeap.size() > k) {
        topKHeap.poll(); // 移除最小的,保持堆中是当前遇到的K个最大元素
    }
}
System.out.println("最大的K个元素: " + topKHeap); // [4, 5, 6] (内部顺序不保证,但poll会按从小到大)

这个例子就很好的说明了,利用

PriorityQueue
的特性,我们能以相对高效的方式解决一些看似复杂的问题。

此外,事件模拟系统霍夫曼编码(Huffman Coding)构建最小生成树等算法中,

PriorityQueue
也扮演着核心角色。它提供了一种高效管理“按优先级处理”元素的机制,让复杂的问题变得清晰和可操作。所以,掌握
PriorityQueue
的使用,对于解决很多实际的编程挑战来说,绝对是一项非常值得投资的技能。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

835

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

740

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

736

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

399

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

43

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.6万人学习

C# 教程
C# 教程

共94课时 | 6.9万人学习

Java 教程
Java 教程

共578课时 | 47.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号