0

0

Java的PriorityQueue怎么用_实现最小堆与最大堆的优先队列

P粉602998670

P粉602998670

发布时间:2026-03-11 10:46:03

|

833人浏览过

|

来源于php中文网

原创

priorityqueue默认是最小堆,队首为最小元素;需传comparator.reverseorder()实现最大堆,自定义类须实现comparable或传comparator,且非线程安全。

java的priorityqueue怎么用_实现最小堆与最大堆的优先队列

PriorityQueue 默认是最小堆,别默认以为它能自动倒序

Java 的 PriorityQueue 默认按自然顺序排序,也就是最小堆:队首(peek())永远是**最小元素**。想让它变成最大堆?不能靠改内部逻辑,得靠传入自定义比较器——这是最常被忽略的前提。

常见错误现象:PriorityQueue<integer> pq = new PriorityQueue(); pq.add(3); pq.add(1); pq.add(2); System.out.println(pq.poll());</integer> 输出 1,有人误以为“没生效”或“要手动排序”,其实它工作完全正常,只是你没意识到它天生就是最小堆。

  • 自然类型(IntegerString 等)直接用空构造函数即可实现最小堆
  • 自定义类必须实现 Comparable 接口,否则运行时抛 ClassCastException
  • 不建议依赖 PriorityQueue 的迭代器顺序——它不保证遍历有序,只保证 poll()/peek() 正确

用 Comparator.reverseOrder() 快速实现最大堆

对基本包装类(如 IntegerDouble),最简方式是传 Comparator.reverseOrder()。它本质是调用 compareTo() 后取反,开销极小,且可读性高。

使用场景:需要频繁取最大值(比如 Top-K 问题、滑动窗口最大值的简化版)、模拟最大堆语义但又不想写完整比较逻辑。

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

Moshi Chat
Moshi Chat

法国AI实验室Kyutai推出的端到端实时多模态AI语音模型,具备听、说、看的能力,不仅可以实时收听,还能进行自然对话。

下载

示例:

PriorityQueue<Integer> maxHeap = new PriorityQueue<>(Comparator.reverseOrder());
maxHeap.offer(3);
maxHeap.offer(1);
maxHeap.offer(2);
System.out.println(maxHeap.poll()); // 输出 3

  • 不要用 new PriorityQueue((a, b) -> b - a) 处理大整数——可能溢出;reverseOrder() 安全
  • 如果元素为 nullreverseOrder() 会抛 NullPointerException,注意判空或用 Comparator.nullsLast() 组合
  • 这个写法在 Java 8+ 可用,低版本需手写匿名类

自定义类怎么进 PriorityQueue:Comparable vs Comparator 二选一

自定义类进 PriorityQueue,必须明确排序依据。要么让类自己实现 Comparable,要么在构造时传 Comparator——二者只能选其一,混用会覆盖前者,且容易引发逻辑混乱。

参数差异明显:new PriorityQueue<someclass>()</someclass> 要求 SomeClass implements Comparable<someclass></someclass>;而 new PriorityQueue(Comparator.comparing(...)) 则完全绕过类本身的 compareTo 方法。

  • 如果类天然有主排序逻辑(如 Task 按优先级排),优先实现 Comparable
  • 如果同一类需在不同场景按不同字段排序(比如一会儿按时间、一会儿按权重),必须用外部 Comparator,避免污染领域模型
  • 实现 Comparable 时,务必保证 compareToequals 一致,否则 contains() 行为可能不符合预期

PriorityQueue 不是线程安全的,多线程下 poll/offer 可能出错

所有操作(offer()poll()peek())都不加锁。并发修改会导致 ConcurrentModificationException 或静默数据错乱——这不是偶发 bug,是设计使然。

性能影响:加同步块(synchronized)会严重拖慢吞吐;用 PriorityBlockingQueue 是更合理的选择,它基于可重入锁 + 条件队列,支持阻塞式操作,且保持堆性质。

  • 别试图用 Collections.synchronizedCollection(new PriorityQueue()) ——它只同步单个方法,无法保证复合操作(如 if (!pq.isEmpty()) pq.poll())原子性
  • PriorityBlockingQueue 不接受 null 元素,这点和 PriorityQueue 一致
  • 如果业务允许“最终一致性”,也可考虑用无锁结构(如 LockFreePriorityQueue 第三方库),但得自行承担复杂度

事情说清了就结束。真正难的不是写对一行 new PriorityQueue(Comparator.reverseOrder()),而是想清楚:这个堆谁在读、谁在写、是否跨线程、排序依据会不会随需求变化——这些才是后期翻车的高发区。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

1010

2023.08.02

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

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

254

2023.09.22

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

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

1089

2024.03.01

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

846

2023.08.22

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

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

334

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

105

2025.10.23

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1902

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

656

2025.10.17

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

3

2026.03.11

热门下载

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

精品课程

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

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11万人学习

Java 教程
Java 教程

共578课时 | 80万人学习

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

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