0

0

在Java里PriorityQueue如何实现优先级队列_Java优先级队列实现解析

P粉602998670

P粉602998670

发布时间:2026-02-07 09:58:04

|

284人浏览过

|

来源于php中文网

原创

PriorityQueue底层基于最小堆(二叉堆)实现,使用动态扩容的Object数组,通过下标关系维护堆序,非红黑树;插入/删除为O(log n),不支持高效查找,遍历无序,需用poll()获取有序序列。

在java里priorityqueue如何实现优先级队列_java优先级队列实现解析

PriorityQueue底层用的是二叉堆,不是红黑树

Java的PriorityQueue默认基于**最小堆**(min-heap)实现,底层是动态扩容的Object数组,通过父/子节点下标关系(i → 2i+1i → 2i+2)维护堆序,不是TreeSet那种红黑树结构。这意味着:

  • 插入和删除时间复杂度都是O(log n),但不支持O(log n)查找任意元素
  • 遍历时(比如用for (E e : pq))不保证按优先级顺序——它只保证poll()peek()返回当前最高优先级元素
  • 如果传入Comparator,必须满足“不修改比较逻辑”的前提;若比较器在队列使用中动态变更,行为未定义

构造时指定Comparator要注意null安全和一致性

自定义优先级最常用方式是传Comparator,但容易踩两个坑:

  • 若元素本身为null,而Comparator没做判空(比如直接调用a.compareTo(b)),会抛NullPointerException
  • 比较逻辑必须满足自反性、传递性、对称性;例如用Integer::compareTo比较int字段是对的,但用a > b ? 1 : -1漏掉a == b分支,会导致堆结构错乱
  • 推荐写法:Comparator.comparingInt(Task::priority).reversed()(最大堆)或(a, b) -> Integer.compare(a.priority, b.priority)

offer()和add()没区别,但poll()和element()异常行为不同

这两个方法语义差异直接影响错误处理策略:

小K直播姬
小K直播姬

全球首款AI视频动捕虚拟直播产品

下载
  • offer(E e)add(E e)PriorityQueue里完全等价,都成功返回true(该队列不设容量上限,不会抛IllegalStateException
  • poll():队列空时返回null,适合需要判空后处理的场景
  • element():队列空时抛NoSuchElementException,适合“必须有值”的断言场景;别误用成peek()(后者才返回null

不能靠迭代器排序,要转List再排序就得重新建队列

很多人想“把PriorityQueue里的元素按优先级全取出来”,结果写new ArrayList(pq)再排序,其实白忙:

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

  • PriorityQueue的迭代器不保证顺序,ArrayList构造函数只是按内部数组当前排列复制,不是按堆序
  • 正确做法:反复poll()直到空,自然得到升序(最小堆)序列;如需降序且保留原队列,先clone()或用stream().sorted()(但注意这已脱离堆逻辑)
  • 注意clone()返回的是浅拷贝,元素引用相同;若元素可变,后续修改会影响队列行为
实际用的时候,最常被忽略的是:**堆序只在每次offer()/poll()后由内部修复,中间状态不可靠**。比如往队列里改了某个元素的字段,却不remove()offer(),那它的位置就永远错位了。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

240

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

600

2024.03.01

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

606

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

551

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

173

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

204

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

403

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

582

2023.08.10

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

2

2026.02.06

热门下载

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

精品课程

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

共23课时 | 3.3万人学习

C# 教程
C# 教程

共94课时 | 8.8万人学习

Java 教程
Java 教程

共578课时 | 59.3万人学习

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

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