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 的内部结构。

Sylius开源电子商务平台
Sylius开源电子商务平台

Sylius开源电子商务平台是一个开源的 PHP 电子商务网站框架,基于 Symfony 和 Doctrine 构建,为用户量身定制解决方案。可管理任意复杂的产品和分类,每个产品可以设置不同的税率,支持多种配送方法,集成 Omnipay 在线支付。功能特点:前后端分离Sylius 带有一个强大的 REST API,可以自定义并与您选择的前端或您的微服务架构很好地配合使用。如果您是 Symfony

下载
  • 假设 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

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

910

2023.08.02

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

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

251

2023.09.22

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

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

988

2024.03.01

string转int
string转int

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

910

2023.08.02

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

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

598

2024.08.29

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

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

294

2025.08.29

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

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

210

2025.08.29

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

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

723

2023.08.10

Swift iOS架构设计与MVVM模式实战
Swift iOS架构设计与MVVM模式实战

本专题聚焦 Swift 在 iOS 应用架构设计中的实践,系统讲解 MVVM 模式的核心思想、数据绑定机制、模块拆分策略以及组件化开发方法。内容涵盖网络层封装、状态管理、依赖注入与性能优化技巧。通过完整项目案例,帮助开发者构建结构清晰、可维护性强的 iOS 应用架构体系。

0

2026.03.03

热门下载

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

精品课程

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

共23课时 | 4.1万人学习

C# 教程
C# 教程

共94课时 | 10.6万人学习

Java 教程
Java 教程

共578课时 | 76万人学习

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

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