面试tips--java--equals() & hashCode()

1. equals() 方法

  • 作用 :用于比较两个对象的 内容是否相等

  • 默认实现 :在 Object 类中,equals() 默认使用 == ,比较的是对象的 内存地址(即是否为同一个对象)。

  • 常见重写场景 :当我们希望比较对象的"内容"而不是"地址"时,就需要重写 equals() 方法。

    • 比如 String 类就重写了 equals(),所以 new String("abc").equals(new String("abc")) 会返回 true,即使它们不是同一个对象。

例子:

java 复制代码
public class Person {
    private String name;
    private int age;

    @Override
    public boolean equals(Object o) {
        if (this == o) return true; // 内存地址相同,直接true
        if (o == null || getClass() != o.getClass()) return false; // 类型不同直接false
        Person person = (Person) o;
        return age == person.age && name.equals(person.name); // 内容比较
    }
}

2. hashCode() 方法

  • 作用 :返回对象的 哈希值int 类型),主要用于 哈希结构的集合 (如 HashMapHashSetHashtable 等)。

  • 哈希表存储原理

    1. 插入对象时,先计算对象的 hashCode()

    2. hashCode() 定位到某个"桶"(bucket)。

    3. 如果桶中有多个对象,再用 equals() 判断对象是否相同。

  • 默认实现 :在 Object 类中,hashCode() 会根据对象的内存地址计算,因此不同对象一般哈希值不同。

例子:

java 复制代码
public class Person {
    private String name;
    private int age;

    @Override
    public int hashCode() {
        return name.hashCode() + age;
    }
}

3. 为什么重写 equals() 必须重写 hashCode()

3.1 Java 规定

Java 规定了 equals()hashCode()契约

  1. 如果两个对象 equals() 返回 true,那么它们的 hashCode() 必须相等。

  2. 如果两个对象 equals() 返回 false,它们的 hashCode() 可以相等(但最好减少冲突)。

3.2 原因

如果只重写了 equals(),没重写 hashCode(),可能导致集合中找不到对象。

示例:

java 复制代码
Person p1 = new Person("Tom", 18);
Person p2 = new Person("Tom", 18);

HashSet<Person> set = new HashSet<>();
set.add(p1);

System.out.println(set.contains(p2));
  • 如果 只重写了 equals()

    • p1.equals(p2)true(内容相等)。

    • p1.hashCode()p2.hashCode() 不同(因为默认按内存地址算)。

    • HashSet 会认为它们在不同的桶中 → contains(p2) 返回 false ❌。

  • 如果 同时重写了 equals()hashCode()

    • 哈希值相同,定位到同一个桶。

    • 再用 equals() 比较内容 → 找到对象 → contains(p2) 返回 true ✅。


4. 正确的重写方式

推荐使用 IDE 自动生成 ,例如 IntelliJ IDEA/ Eclipse 可以自动生成基于字段的 equals()hashCode()

示例(推荐写法)

java 复制代码
import java.util.Objects;

public class Person {
    private String name;
    private int 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);
    }
}

5. 总结

  1. equals():比较对象的内容是否相等。

  2. hashCode():返回对象的哈希值,用于哈希集合快速查找。

  3. 必须遵守契约

    • equals() 相等 → hashCode() 必须相等。

    • 否则哈希集合(HashMap/HashSet)会出现"存了却找不到"的问题。

  4. 正确姿势 :重写 equals() 时,一定要重写 hashCode()

相关推荐
独断万古他化1 小时前
【Spring 原理】Bean 的作用域与生命周期
java·后端·spring
m0_694845571 小时前
tinylisp 是什么?超轻量 Lisp 解释器编译与运行教程
服务器·开发语言·云计算·github·lisp
*小海豚*1 小时前
在linux服务器上DNS正常,但是java应用调用第三方解析域名报错
java·linux·服务器
春日见1 小时前
如何创建一个PR
运维·开发语言·windows·git·docker·容器
C++ 老炮儿的技术栈1 小时前
VS2015 + Qt 实现图形化Hello World(详细步骤)
c语言·开发语言·c++·windows·qt
派葛穆2 小时前
Python-批量安装依赖
开发语言·python
撩得Android一次心动2 小时前
Android LiveData 全面解析:使用Java构建响应式UI【源码篇】
android·java·android jetpack·livedata
组合缺一2 小时前
Solon AI (Java) v3.9 正式发布:全能 Skill 爆发,Agent 协作更专业!仍然支持 java8!
java·人工智能·ai·llm·agent·solon·mcp
MSTcheng.2 小时前
【C++】C++11新特性(二)
java·开发语言·c++·c++11
晓13132 小时前
第七章 【C语言篇:文件】 文件全面解析
linux·c语言·开发语言