一文带你搞懂HashSet和TreeSet的区别!!!

在Java中,HashSetTreeSet是两种常用的Set集合实现,核心区别在于底层数据结构、元素顺序性和性能特性。本文将从以下几个方面进行详细的对比:

一、底层实现与数据结构

特性 HashSet TreeSet
底层结构 基于HashMap实现(哈希表) 基于TreeMap实现(红黑树)
节点类型 哈希桶(数组+链表/红黑树) 平衡二叉搜索树(自平衡红黑树)
数据分布 元素分散存储,哈希冲突时形成链表/树 元素按排序规则组织为树结构

二、元素顺序性

特性 HashSet TreeSet
顺序保证 不保证顺序(插入顺序可能随机) 自然排序自定义Comparator排序
遍历顺序 无序(依赖哈希函数和扩容机制) 按升序或定义的顺序遍历

三、性能对比

操作 HashSet TreeSet
插入/删除 O(1)(均摊时间,哈希冲突时可能退化) O(log n)(树的高度决定)
查找 O(1)(哈希表直接定位) O(log n)(二分搜索树路径)
遍历 O(n)(顺序不确定) O(n)(中序遍历,按排序顺序)

四、对元素的要求

特性 HashSet TreeSet
哈希码要求 必须正确实现hashCode()equals() 无需hashCode(),但需可比较性
比较规则 依赖equals()判断重复 依赖compareTo()Comparator判断重复
Null支持 允许存储null 不允许null(除非自定义Comparator允许)

五、初始化与扩容

特性 HashSet TreeSet
初始化参数 初始容量(默认16)、负载因子(默认0.75) 无需容量参数(树动态调整)
扩容机制 容量翻倍(2倍),重新哈希所有元素 无扩容,树结构自动平衡

六、适用场景

场景 HashSet TreeSet
高频插入/删除 优先选择(更快的时间复杂度) 适用于需要排序的中等规模数据
快速查找 适合基于哈希的快速访问 适合范围查询(如查找大于某值的元素)
数据去重 默认选择(无序去重) 需要有序去重时使用

七、代码示例与注意事项

1. 自定义对象使用HashSet

必须正确覆盖hashCode()equals()

typescript 复制代码
java
class Student {
    String id;
    String name;

    @Override
    public int hashCode() {
        return Objects.hash(id, name);
    }

    @Override
    public boolean equals(Object obj) {
        if (this == obj) return true;
        if (obj == null || getClass() != obj.getClass()) return false;
        Student other = (Student) obj;
        return Objects.equals(id, other.id) && Objects.equals(name, other.name);
    }
}

// 使用示例
Set<Student> hashSet = new HashSet<>();
hashSet.add(new Student("001", "Alice"));

2. 自定义对象使用TreeSet

需实现Comparable或提供Comparator

typescript 复制代码
java
class Product implements Comparable<Product> {
    String name;
    double price;

    @Override
    public int compareTo(Product other) {
        return Double.compare(this.price, other.price); // 按价格排序
    }
}

// 使用示例
Set<Product> treeSet = new TreeSet<>();
treeSet.add(new Product("Laptop", 999.99));

八、如何选择?

  • 选择HashSet
    需要快速插入、删除和查找,且不关心元素顺序;允许存储null;数据量大但无需排序。
  • 选择TreeSet
    需要元素按自然顺序或自定义顺序遍历;频繁执行范围查询(如subSet()tailSet());数据量中等且需要排序。

通过理解两者的底层实现和特性差异,可以根据实际需求选择最合适的集合实现,从而在性能和功能之间取得最佳平衡。

相关推荐
躺平大鹅17 分钟前
Java面向对象入门(类与对象,新手秒懂)
java
初次攀爬者1 小时前
RocketMQ在Spring Boot上的基础使用
java·spring boot·rocketmq
花花无缺1 小时前
搞懂@Autowired 与@Resuorce
java·spring boot·后端
Derek_Smart3 小时前
从一次 OOM 事故说起:打造生产级的 JVM 健康检查组件
java·jvm·spring boot
NE_STOP4 小时前
MyBatis-mybatis入门与增删改查
java
孟陬7 小时前
国外技术周刊 #1:Paul Graham 重新分享最受欢迎的文章《创作者的品味》、本周被划线最多 YouTube《如何在 19 分钟内学会 AI》、为何我不
java·前端·后端
想用offer打牌7 小时前
一站式了解四种限流算法
java·后端·go
阿巴斯甜7 小时前
Android 报错:Zip file '/Users/lyy/develop/repoAndroidLapp/l-app-android-ble/app/bu
android
华仔啊7 小时前
Java 开发千万别给布尔变量加 is 前缀!很容易背锅
java
Kapaseker7 小时前
实战 Compose 中的 IntrinsicSize
android·kotlin