Java 中 PriorityQueue 的底层数据结构及相关分析

Java 中 PriorityQueue 的底层数据结构及相关分析

1. PriorityQueue 的底层数据结构

在 Java 中,PriorityQueue 的底层数据结构是基于堆(Heap)实现的二叉堆(Binary Heap),默认使用最小堆(Min Heap),即堆顶元素是队列中的最小元素。

PriorityQueue 采用数组(Object[])**来存储数据,并使用**二叉堆结构来维护优先级关系。

2. PriorityQueue 的实现原理

2.1 插入(offer/add)

  • 新元素插入到堆的末尾(数组的最后一个位置)。
  • 通过**上浮(siftUp)**操作,确保堆的有序性。
  • 上浮操作通过比较当前节点与其父节点的大小,若当前节点比父节点小,则交换它们,直到满足最小堆的性质。

2.2 删除(poll/remove)

  • 删除堆顶元素(最小元素)。
  • 用堆的最后一个元素填补堆顶位置。
  • 通过**下沉(siftDown)**操作,确保堆的有序性。
  • 下沉操作通过比较当前节点与其子节点的大小,若当前节点大于子节点,则交换它们,直到满足最小堆的性质。

2.3 获取堆顶元素(peek)

  • 直接返回数组的第一个元素(堆顶)。
  • 时间复杂度 O(1)

2.4 复杂度分析

  • 插入(offer/add):O(log n)
  • 删除(poll/remove):O(log n)
  • 访问堆顶元素(peek):O(1)

3. PriorityQueue 的应用场景

  1. 任务调度(按优先级处理任务)
  2. Dijkstra 最短路径算法
  3. Top K 问题(获取前 K 个最大/最小元素)
  4. 流式数据处理(如实时求中位数)
  5. 操作系统进程调度(按照优先级处理不同任务)
  6. 数据合并(如合并多个有序数组)

4. PriorityQueue 的优缺点

4.1 优点

  • 自动维护有序性,取最小/最大元素效率高。
  • 插入、删除操作高效,时间复杂度 O(log n)
  • 基于堆的高效实现,适用于大量数据的优先级处理。

4.2 缺点

  • 不支持随机访问,不能像 ArrayList 一样通过索引直接访问。
  • 只保证堆顶元素是最小/最大,无法保证整体有序。
  • 非线程安全,如需多线程使用,可考虑 PriorityBlockingQueue

5. 替代方案

需求替代方案
需要线程安全的优先队列PriorityBlockingQueue
需要基于平衡二叉树的优先队列TreeSet(红黑树)
需要维护多个优先级队列ConcurrentSkipListSet(跳表)
需要自定义排序规则TreeMap

6. PriorityQueue 使用示例

6.1 基本用法

import java.util.PriorityQueue;public class PriorityQueueExample {public static void main(String[] args) {// 创建最小堆PriorityQueue<Integer> pq = new PriorityQueue<>();pq.offer(5);pq.offer(1);pq.offer(3);pq.offer(7);pq.offer(2);// 依次取出最小元素while (!pq.isEmpty()) {System.out.println(pq.poll());}}
}

输出:

1
2
3
5
7

6.2 自定义排序(最大堆)

import java.util.Collections;
import java.util.PriorityQueue;public class MaxHeapExample {public static void main(String[] args) {PriorityQueue<Integer> maxHeap = new PriorityQueue<>(Collections.reverseOrder());maxHeap.offer(5);maxHeap.offer(1);maxHeap.offer(3);maxHeap.offer(7);maxHeap.offer(2);while (!maxHeap.isEmpty()) {System.out.println(maxHeap.poll());}}
}

输出:

7
5
3
2
1

6.3 自定义对象排序

import java.util.PriorityQueue;
import java.util.Comparator;class Task {String name;int priority;public Task(String name, int priority) {this.name = name;this.priority = priority;}@Overridepublic String toString() {return name + " (Priority: " + priority + ")";}
}public class TaskScheduler {public static void main(String[] args) {PriorityQueue<Task> taskQueue = new PriorityQueue<>(Comparator.comparingInt(t -> t.priority));taskQueue.offer(new Task("Task A", 3));taskQueue.offer(new Task("Task B", 1));taskQueue.offer(new Task("Task C", 2));while (!taskQueue.isEmpty()) {System.out.println(taskQueue.poll());}}
}

输出:

Task B (Priority: 1)
Task C (Priority: 2)
Task A (Priority: 3)

7. 总结

  1. 底层结构:基于二叉堆,使用数组存储。
  2. 操作复杂度:插入/删除 O(log n),访问 O(1)
  3. 应用场景:任务调度、最短路径、Top K 等。
  4. 优缺点:自动排序但不支持随机访问,非线程安全。
  5. 替代方案:线程安全用 PriorityBlockingQueue,基于红黑树用 TreeSet

PriorityQueue 是 Java 中处理优先级任务的高效工具,在需要动态维护最小或最大元素的场景下非常有用。

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

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

相关文章

springboot实现调用百度ocr实现身份识别+二要素校验

一、技术选型 OCR服务&#xff1a;推荐使用百度AI 二、实现 1.注册一个服务 百度智能云控制台https://console.bce.baidu.com/ai-engine/ocr/overview/index?_1742309417611 填写完之后可以获取到app-id、apiKey、SecretKey这三个后面文件配置会用到 2、导入依赖 <!-- …

【数据分享】2000—2024年我国乡镇的逐月归一化植被指数(NDVI)数据(Shp/Excel格式)

之前我们分享过2000—2024年我国省市县三级逐月归一化植被指数&#xff08;NDVI&#xff09;数据&#xff0c;该数据是基于NASA定期发布的MOD13A3数据集中的月度NDVI栅格数据&#xff08;可查看之前的文章获悉详情&#xff09;计算得出。很多小伙伴拿到数据后反馈是否可以处理出…

背包问题——动态规划的经典问题包括01背包问题和完全背包问题

01背包问题&#xff1a;给你多个物品每个物品只能选一次&#xff0c;要你在不超过背包容积&#xff08;或者恰好等于&#xff09;的情况下选择装价值最大的组合。如果没有动态规划的基础其实是很难理解这个问题的&#xff0c;所以看这篇文章之前先去学习一下动态规划的基本思想…

AI Agent系列(七) -思维链(Chain of Thought,CoT)

AI Agent系列【七】 前言一、CoT技术详解1.1 CoT组成1.2 CoT的特点 二、CoT的作用三、CoT的好处四、CoT适用场景五、CoT的推理结构 前言 思维链(Chain of Thought,CoT)&#xff0c;思维链就是一系列中间的推理步骤(a series of intermediate reasoning steps)&#xff0c;通过…

Docker搭建Testlink教程

1.拉取镜像 打开终端输入命令&#xff1a; #拉取mariadb镜像 docker pull bitnami/mariadb #拉取testlink镜像 docker pull bitnami/testlink-archived 执行结果&#xff1a; 2.运行容器 打开终端输入命令&#xff1a; #创建容器网络 docker network create testlink #查…

考研c语言复习之栈

栈一般出选择题&#xff0c;队列选择题和大题都有 栈&#xff1a;只允许在一端 进行插入或删除操作的线性表即栈顶&#xff08;top) s.top-1时栈为空 向栈中插入元素 s.tops.top1;s.data[s.top]value; 这段代码可以用一行代码代替&#xff1a; s.data[s.top]value; 不懂i和…

C#里使用libxl来合并单元格的例子

操作EXCEL的文件格式是常用的功能&#xff0c; 通过不同的单元格的合并&#xff0c;可以生成不同的表格。 如下图所示&#xff1a; 采用libxl来创建上面的EXCEL&#xff0c;使用下面的代码来实现&#xff1a; private void button8_Click(object sender, EventArgs e) {var …

大屏技术汇集【目录】

Cesium 自从首次发布以来&#xff0c;经历了多个版本的迭代和更新&#xff0c;每个版本都带来了性能改进、新功能添加以及对现有功能的优化。以下是 Cesium 一些重要版本及其主要特点&#xff1a; 主要版本概述 Cesium 1.0 (2012年) 初始版本发布&#xff0c;确立了Cesium作为…

《深度学习》——YOLOv3详解

文章目录 YOLOv3简介YOLOv3核心原理YOLOv3改进YOLOv3网络结构 YOLOv3简介 YOLOv3&#xff08;You Only Look Once, version 3&#xff09;是一种先进的实时目标检测算法&#xff0c;由 Joseph Redmon 和 Ali Farhadi 开发。它在目标检测领域表现出色&#xff0c;具有速度快、精…

websocket中spring注入失效

一个null指针引发的思考 websocket中spring注入失效 一个null指针引发的思考场景代码SpringBoot入口类配置类websocket类 问题排查问题1&#xff1a;问题2&#xff1a; 反思解决方案一&#xff1a;方案二&#xff1a;方案三&#xff1a;方案四&#xff1a; 场景 首页有个webso…

QT开发(4)--各种方式实现HelloWorld

目录 1. 编辑框实现 2. 按钮实现 前面已经写过通过标签实现的了&#xff0c;所以这里就不写了&#xff0c;通过这两个例子&#xff0c;其他的也是同理 1. 编辑框实现 编辑框分为单行编辑框&#xff08;QLineEdit&#xff09;双行编辑框&#xff08;QTextEdit&#xff09;&am…

自由学习记录(45)

顶点片元着色器&#xff08;important&#xff09; 1.需要在Pass渲染通道中编写着色器逻辑 2.可以使用cG或HLSL两种shader语言去编写Shader逻辑 3.代码量较多&#xff0c;灵活性较强&#xff0c;性能消耗更可控&#xff0c;可以实现更多渲染细节 4.适用于光照处理较少&#xf…

内存管理(C++篇)

前言 我们在C语言阶段学习过内存管理的相关操作和知识&#xff0c;比如说malloc&#xff0c;calloc等内存开辟函数&#xff0c;但我们在学的时候会发现&#xff0c;使用这些函数还是相对来说比较冗杂的&#xff0c;那么今天我们来学习C语言中相关的内存管理操作&#xff0c;相信…

母婴电商企业案例:日事清驱动项目管理执行与OKR目标管理的流程自动化实践

一、关于科木电商 “小鹿豆豆”&#xff0c;一个年轻的品牌&#xff0c;近期在无论是淘宝、拼多多还是抖音电商平台&#xff0c;都成了亮眼的爆品。这个由绵阳科木电子商务有限公司推出的新品牌&#xff0c;以其高品质的保湿云柔巾迅速赢得了母婴护理市场的青睐&#xff0c;特别…

图数据库Neo4j和JDK安装与配置教程(超详细)

目录 前言 一、Java环境配置 &#xff08;一&#xff09;JDK的下载与安装 &#xff08;二&#xff09;JDK环境配置 &#xff08;三&#xff09;检测JDK17是否配置成功 二、Neo4j的安装与配置 &#xff08;一&#xff09;Neo4j的下载与安装 &#xff08;二&#xff09;N…

git原理与常用命令及其使用

认识工作区、暂存区、版本库 ⼯作区&#xff1a;是在电脑上你要写代码或⽂件的⽬录。 暂存区&#xff1a;英⽂叫 stage 或 index。⼀般存放在 .git ⽬录下的 index ⽂件&#xff08;.git/index&#xff09;中&#xff0c;我们 把暂存区有时也叫作索引&#xff08;index&#xf…

Web-Machine-N7靶机通关攻略

获取靶机ip arp-scan -l 端口扫描 nmap xxxx 访问80端口发现没用 扫描目录 gobuster dir -u http:/192.168.117.160 -w /usr/share/wordlists/dirbuster/directory-list-2.3-medium,txt -x php,html,txt ,zip 打开exploit.html 点击F12&#xff0c;修改localhost为靶机ip&#…

2025-03-21 Unity 网络基础3——TCP网络通信准备知识

文章目录 1 IP/端口类1.1 IPAddress1.2 IPEndPoint 2 域名解析2.1 IPHostEntry2.2 Dns 3 序列化与反序列化3.1 序列化3.1.1 内置类型 -> 字节数组3.1.2 字符串 -> 字节数组3.1.3 类对象 -> 字节数组 3.2 反序列化3.2.1 字节数组 -> 内置类型3.2.2 字节数组 -> 字…

Java-servlet(七)详细讲解Servlet注解

Java-servlet&#xff08;七&#xff09;详细讲解Servlet注解 前言一、注解的基本概念二、Override 注解2.1 作用与优势2.2 示例代码 三、Target 注解3.1 定义与用途3.2 示例代码 四、WebServlet 注解4.1 作用4.2 示例代码 五、反射与注解5.1 反射的概念5.2 注解与反射的结合使…

nginx 反向代理 ubuntu

关键字 Nginx&#xff0c;正向代理&#xff0c;方向代理&#xff0c;博客建站 背景环境 我在搭建个人博客的过程中遇到一个问题&#xff0c;我的博客服务的端口是1313&#xff0c;我的域名是qinyangx.top。我希望能够通过qinyangx.top直接访问到服务器上1313端口的博客服务。…