0

0

Java集合框架中的Map集合与自定义类型

P粉602998670

P粉602998670

发布时间:2026-01-13 13:54:42

|

282人浏览过

|

来源于php中文网

原创

Map的key必须重写equals()和hashCode(),因哈希表靠hashCode()定位桶、equals()判断键等价;若不重写,默认地址比较会导致相同逻辑对象被误判为不同key。

java集合框架中的map集合与自定义类型

Map 的 key 为什么必须重写 equals()hashCode()

因为 HashMapLinkedHashMap 等基于哈希表的实现,靠 hashCode() 定位桶位置,再用 equals() 判定是否为同一 key。若不重写,所有自定义对象默认继承 Object 的实现,即地址比较——两个内容相同的对象也会被当作不同 key 存入。

  • 没重写时,map.put(new Person("Alice", 25), "A")map.get(new Person("Alice", 25)) 一定返回 null
  • hashCode() 不一致会导致 key 散列到不同桶,equals() 就根本不会被调用
  • 只要字段参与逻辑相等判断(比如业务上认为 name+age 相同即为同一人),这些字段就必须同时用于 hashCode()equals()

使用 Lombok 自动生成时要注意什么

@Data 看似省事,但会无差别地把所有字段纳入 equals()/hashCode() 计算,可能引入隐含 bug。

  • 如果类里有瞬态字段(如 transient String cache)、运行时计算字段(如 long lastAccessTime)或数据库主键 ID(新增对象 ID 为 null,但逻辑上应视为相同),它们不该参与比较
  • 推荐显式用 @EqualsAndHashCode(onlyExplicitlyIncluded = true) + @EqualsAndHashCode.Include 标注真正需要参与比较的字段
  • Lombok 生成的 hashCode() 是按字段顺序累加哈希值,字段顺序变化会影响结果(虽不常见,但在跨版本序列化或缓存场景下需留意)

作为 Map key 的自定义类型能否修改?

能改,但改完就很可能再也取不到它了——除非你同步更新整个 Map 的内部结构。

FloatSearch
FloatSearch

FloatSearch是一个专业的AI搜索引擎,提供多样化的见解

下载
  • 假设 Person p = new Person("Bob", 30) 已作为 key 存入 HashMap,之后执行 p.setAge(31):其 hashCode() 变了,但 Map 并不知道,仍去旧桶里找,get(p) 返回 null
  • 即使你记得手动 remove()put(),多线程下仍可能因竞态导致丢失数据
  • 最佳实践是让 key 类型不可变(final 字段 + 无 setter),例如用 record Person(String name, int age) { } —— Java 14+ 原生支持,自动实现安全的 equals()/hashCode()

TreeMap 对自定义 key 的要求完全不同

TreeMap 不依赖哈希,而是靠 Comparable 或外部 Comparator 排序,所以它根本不看 hashCode(),但强制要求 key 可比较。

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

  • 若 key 类未实现 Comparable,又没传 Comparator,构造 TreeMap 时不会报错,但第一次 put() 就抛 ClassCastException: xxx cannot be cast to java.lang.Comparable
  • 实现 Comparable 时,compareTo() 必须与 equals() 保持一致:若 a.equals(b) 为 true,则 a.compareTo(b) == 0;否则 TreeMap 可能违反集合契约(比如 containsKey() 返回 false,但遍历时又能遍历到该 key)
  • Comparator 更灵活,可复用已有比较逻辑,但要注意:同一个 TreeMap 实例不能切换比较器,必须在构造时确定
public class Person implements Comparable<Person> {
    private final String name;
    private final int age;

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    @Override
    public int compareTo(Person o) {
        int nameCmp = this.name.compareTo(o.name);
        if (nameCmp != 0) return nameCmp;
        return Integer.compare(this.age, o.age);
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        Person person = (Person) o;
        return age == person.age && Objects.equals(name, person.name);
    }

    @Override
    public int hashCode() {
        return Objects.hash(name, age);
    }
}
实际用的时候,别只盯着“能跑”,得想清楚这个 key 是进哈希表还是红黑树,改不改、谁来管一致性——这些细节一旦漏掉,问题往往出现在上线后查半天才定位到那一行没加 final 的字段。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

1051

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的相关内容,可以阅读本专题下面的文章。

1131

2024.03.01

string转int
string转int

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

1051

2023.08.02

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

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

617

2024.08.29

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

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

335

2025.08.29

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

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

235

2025.08.29

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

786

2023.08.10

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

69

2026.03.13

热门下载

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

精品课程

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

共23课时 | 4.5万人学习

C# 教程
C# 教程

共94课时 | 11.4万人学习

Java 教程
Java 教程

共578课时 | 82.9万人学习

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

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