0

0

在Java中TreeSet如何实现排序_Java排序集合底层机制解析

P粉602998670

P粉602998670

发布时间:2026-02-03 09:17:02

|

870人浏览过

|

来源于php中文网

原创

TreeSet基于红黑树实现,插入即有序,时间复杂度O(log n),依赖Comparable或Comparator排序,去重依据比较结果为0而非equals(),不支持随机访问。

在java中treeset如何实现排序_java排序集合底层机制解析

TreeSet 默认按自然顺序排序,本质是红黑树维护的有序结构

TreeSet 不是简单调用 Arrays.sort() 那类临时排序,它从插入第一元素起就维持整体有序。底层用的是 TreeMap 实例——所有元素作为 key 存入,value 固定为 PRESENT(一个空对象)。这意味着每次 add() 都触发红黑树的插入+自平衡操作,时间复杂度稳定在 O(log n),而非数组排序的 O(n log n)

关键点在于:TreeSet 本身不重写比较逻辑,它完全依赖元素是否实现 Comparable 接口,或构造时传入 Comparator。没提供且元素不可比,运行时抛 ClassCastException,不是编译报错。

自定义排序必须显式传 Comparator,不能靠重写 toString 或 equals

常见误区是以为重写 toString()equals() 能影响 TreeSet 排序——完全无效。排序唯一入口是 ComparatorComparable.compareTo()

  • 若元素类已实现 Comparable(如 StringInteger),直接 new TreeSet() 即可
  • 若想按不同规则排序(比如 Person 按年龄而非姓名),必须传 Comparator 实例,例如:
    TreeSet set = new TreeSet<>((p1, p2) -> Integer.compare(p1.getAge(), p2.getAge()));
  • Lambda 中避免直接用 p1.getAge() - p2.getAge(),整数溢出会导致错误排序(如 -2000000000 和 2000000000 相减越界)

add() 失败不报错,但 size 不变——这是去重逻辑在生效

TreeSet 是 Set,天然去重。判断“重复”的依据不是 equals(),而是比较结果为 0:compare(a, b) == 0a.compareTo(b) == 0。这点和 HashSet 完全不同。

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

Glarity
Glarity

Glarity是一款免费开源的AI浏览器扩展,提供YouTube视频总结、网页摘要、写作工具等功能,支持免费的镜像翻译,电子邮件写作辅助,AI问答等功能。

下载

因此可能出现:两个对象 equals() 返回 false,但因 compareTo() 返回 0,被 TreeSet 当作同一元素拒绝插入。

  • 调试时发现 add() 返回 falsesize() 没变,优先检查比较逻辑是否把不该等价的对象判为等价
  • 如果业务上需要保留“内容不同但排序值相同”的对象(比如多个同分数学生),TreeSet 不适用,该换 PriorityQueue 或手动维护 List + Collections.sort()
  • 注意 Comparator.nullsFirst() / nullsLast() 的使用场景——当字段可能为 null 时,不包装会直接抛 NullPointerException

遍历顺序永远有序,但迭代器不支持快速随机访问

TreeSet 的 iterator() 返回的是红黑树中序遍历结果,所以 for-eachstream() 都天然升序。但别试图用索引取第 N 个元素——它没有 get(int index) 方法,也没有基于数组的随机访问能力。

若需按排名查元素(如“分数第 3 高的学生”),要么转成 ArrayList 再取索引(代价 O(n)),要么用 TreeSethigher()lower()ceiling() 等导航方法逼近,但这些只适合找邻近值,不适合精确排名。

真正需要频繁按秩访问的场景,TreeSet 就不是最优解——这时候该考虑 Apache Commons CollectionsTreeList,或自己封装带 rank 维护的结构。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

523

2023.08.02

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

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

238

2023.09.22

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

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

519

2024.03.01

sort排序函数用法
sort排序函数用法

sort排序函数的用法:1、对列表进行排序,默认情况下,sort函数按升序排序,因此最终输出的结果是按从小到大的顺序排列的;2、对元组进行排序,默认情况下,sort函数按元素的大小进行排序,因此最终输出的结果是按从小到大的顺序排列的;3、对字典进行排序,由于字典是无序的,因此排序后的结果仍然是原来的字典,使用一个lambda表达式作为key参数的值,用于指定排序的依据。

396

2023.09.04

string转int
string转int

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

523

2023.08.02

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

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

547

2024.08.29

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

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

153

2025.08.29

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

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

201

2025.08.29

AO3官网入口与中文阅读设置 AO3网页版使用与访问
AO3官网入口与中文阅读设置 AO3网页版使用与访问

本专题围绕 Archive of Our Own(AO3)官网入口展开,系统整理 AO3 最新可用官网地址、网页版访问方式、正确打开链接的方法,并详细讲解 AO3 中文界面设置、阅读语言切换及基础使用流程,帮助用户稳定访问 AO3 官网,高效完成中文阅读与作品浏览。

89

2026.02.02

热门下载

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

精品课程

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

共23课时 | 3.2万人学习

C# 教程
C# 教程

共94课时 | 8.4万人学习

Java 教程
Java 教程

共578课时 | 56.3万人学习

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

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