贝利信息

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

日期:2026-01-13 00:00 / 作者:P粉602998670
Map的key必须重写equals()和hashCode(),因哈希表靠hashCode()定位桶、equals()判断键等价;若不重写,默认地址比较会导致相同逻辑对象被误判为不同key。

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

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

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

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

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

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

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

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

public class Person implements Comparable {
    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 的字段。