当前位置:首页 > 文章列表 > 文章 > java教程 > JavaPriorityQueue使用详解与实战教程

JavaPriorityQueue使用详解与实战教程

2025-07-22 18:57:50 0浏览 收藏

掌握Java优先队列PriorityQueue,提升程序效率!本文详细介绍了Java中PriorityQueue的用法,这是一种基于堆实现的特殊队列,能自动按优先级排序元素。从构造函数(默认容量、指定容量、自定义比较器、集合初始化)到常用方法(offer/add、poll/remove、peek、size、contains),逐一解析其功能与特性。通过示例代码,演示如何使用PriorityQueue实现小根堆、大根堆以及自定义对象排序。同时,对比PriorityQueue与TreeSet的区别,助你根据实际需求选择更合适的集合类,优化你的Java代码。深入理解PriorityQueue的时间复杂度、自定义排序规则等关键知识点,让你的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 Java优先队列用法详解

Java中的PriorityQueue是一种特殊的队列,它允许我们按照元素的优先级来处理它们。简单来说,你可以把它想象成一个自动排序的队列,每次取出的是优先级最高的元素。

如何在Java中使用PriorityQueue Java优先队列用法详解

PriorityQueue的本质是一个堆(通常是小根堆),这意味着队列中的元素总是按照某种顺序排列,保证队首元素是最小(或最大,取决于你的比较器)。

解决方案

使用PriorityQueue的关键在于理解它的构造函数和常用方法。

如何在Java中使用PriorityQueue Java优先队列用法详解
  1. 构造函数:

    • PriorityQueue():创建一个具有默认初始容量(11)的 PriorityQueue,并根据其元素的自然顺序对其进行排序。
    • PriorityQueue(int initialCapacity):创建一个具有指定初始容量的 PriorityQueue,并根据其元素的自然顺序对其进行排序。
    • PriorityQueue(Comparator comparator):创建一个具有默认初始容量的 PriorityQueue,并根据指定的比较器对其元素进行排序。
    • PriorityQueue(int initialCapacity, Comparator comparator):创建一个具有指定初始容量的 PriorityQueue,并根据指定的比较器对其元素进行排序。
    • PriorityQueue(Collection c):创建一个包含指定集合中的元素的 PriorityQueue。
    • PriorityQueue(PriorityQueue c):创建一个包含指定优先级队列中的元素的 PriorityQueue。
    • PriorityQueue(SortedSet c):创建一个包含指定排序集合中的元素的 PriorityQueue。
  2. 常用方法:

    如何在Java中使用PriorityQueue Java优先队列用法详解
    • 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 的区别是什么?

PriorityQueueTreeSet 都是用于存储有序元素的集合,但它们在实现和使用上有一些关键区别:

  • 底层数据结构: PriorityQueue 基于堆(通常是小根堆)实现,而 TreeSet 基于红黑树实现。
  • 排序方式: PriorityQueue 只保证队首元素是最小(或最大)的,其他元素的顺序不确定。 TreeSet 则保证所有元素都是有序的。
  • 是否允许 null 元素: PriorityQueue 不允许存储 null 元素,会抛出 NullPointerExceptionTreeSet 也不允许存储 null 元素(在大多数实现中)。
  • 时间复杂度: PriorityQueueofferpoll 操作的时间复杂度为 O(log n),peek 操作为 O(1)。 TreeSetaddremovecontains 操作的时间复杂度都为 O(log n)。
  • 应用场景: PriorityQueue 适用于需要频繁获取最小(或最大)元素的场景,例如任务调度、堆排序等。 TreeSet 适用于需要维护有序集合的场景,例如字典、索引等。

简单来说,如果只需要获取优先级最高的元素,使用 PriorityQueue 效率更高。如果需要维护一个完全有序的集合,使用 TreeSet 更合适。

文中关于java,时间复杂度,堆,comparator,PriorityQueue的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《JavaPriorityQueue使用详解与实战教程》文章吧,也可关注golang学习网公众号了解相关技术文章。

AI视频制作全流程解析:从画面到配音AI视频制作全流程解析:从画面到配音
上一篇
AI视频制作全流程解析:从画面到配音
Java线程池类型及使用场景解析
下一篇
Java线程池类型及使用场景解析
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    511次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    498次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • AI歌曲生成器:免费在线创作,一键生成原创音乐
    AI歌曲生成器
    AI歌曲生成器,免费在线创作,简单模式快速生成,自定义模式精细控制,多种音乐风格可选,免版税商用,让您轻松创作专属音乐。
    17次使用
  • MeloHunt:免费AI音乐生成器,零基础创作高品质音乐
    MeloHunt
    MeloHunt是一款强大的免费在线AI音乐生成平台,让您轻松创作原创、高质量的音乐作品。无需专业知识,满足内容创作、影视制作、游戏开发等多种需求。
    17次使用
  • 满分语法:免费在线英语语法检查器 | 论文作文邮件一键纠错润色
    满分语法
    满分语法是一款免费在线英语语法检查器,助您一键纠正所有英语语法、拼写、标点错误及病句。支持论文、作文、翻译、邮件语法检查与文本润色,并提供详细语法讲解,是英语学习与使用者必备工具。
    27次使用
  • 易销AI:跨境电商AI营销专家 | 高效文案生成,敏感词规避,多语言覆盖
    易销AI-专为跨境
    易销AI是专为跨境电商打造的AI营销神器,提供多语言广告/产品文案高效生成、精准敏感词规避,并配备定制AI角色,助力卖家提升全球市场广告投放效果与回报率。
    28次使用
  • WisFile:免费AI本地文件批量重命名与智能归档工具
    WisFile-批量改名
    WisFile是一款免费AI本地工具,专为解决文件命名混乱、归类无序难题。智能识别关键词,AI批量重命名,100%隐私保护,让您的文件井井有条,触手可及。
    28次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码