Java面试题-为什么重写equals就一定要重写hashCode方法呢?

news2025/7/9 17:56:26

目录

1、为什么要重写equals 方法

2、hashCode 方法

3、为什么要一起重写?

4 原因分析

总结


先放结论:

  • hashCode 和 equals 两个方法是用来协同判断两个对象是否相等的,采用这种方式的原因是可以提高程序插入和查询的速度。
  • 如果只重写equals方法,不重写hashCode方法,就有可能导致a.equals(b)这个表达式成立,但是hashCode却不同。会造成一个完全相同的对象会存储在hash表的不同位置。

1、为什么要重写equals 方法

Object 类中的 equals 方法用于检测一个对象是否等于另外一个对象。在 Object 类中,这个方法将判断两个对象是否具有相同的引用。如果两个对象具有相同的引用,它们一定是相等的。​

public boolean equals(Object obj) {
        return (this == obj);
    }

通过上述源码和 equals 的定义我们可以看出,在大多数情况来说,equals 的判断是没有什么意义的!例如,使用 Object 中的 equals 比较两个自定义的对象是否相等这就完全没有意义(因为无论对象是否相等,结果都是 false)。 只要不是同一个引用对象就只能是false。

通过以下示例,就可以说明这个问题: 

package cn.itcast.interview;

/**
 * @author: msf
 * @date: 2022/11/26
 */
public class EqualsMyClassExample {
    public static void main(String[] args) {
        Person u1 = new Person();
        u1.setName("Java");
        u1.setAge(18);

        Person u2 = new Person();
        u2.setName("Java");
        u2.setAge(18);

        System.out.println(u1.equals(u2));
    }
}

class Person{
    private String name;
    private int age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
}

 上面代码结果

 因此通常情况下,我们要判断两个对象是否相等,一定要重写 equals 方法,这就是为什么要重写 equals 方法的原因。

2、hashCode 方法

hashCode 翻译为中文是散列码,它是由对象推导出的一个整型值,并且这个值为任意整数,包括正数或负数。​

需要注意的是:散列码是没有规律的。

  • 如果 x 和 y 是两个不同的对象,x.hashCode() 与 y.hashCode() 基本上不会相同;但是也有例外
  • 但如果 a 和 b 相等,则 a.hashCode() 一定等于 b.hashCode()。​

hashCode 使用

package cn.itcast.interview;

/**
 * @author: msf
 * @date: 2022/11/26
 */
public class HashCodeExample {
    public static void main(String[] args) {
        String s1 = "Hello";
        String s2 = "Hello";
        String s3 = "Java";
        System.out.println("s1 hashCode" + s1.hashCode());
        System.out.println("s1 hashCode" + s2.hashCode());
        System.out.println("s1 hashCode" + s3.hashCode());

        // 在一些特殊情况下是相同的
        String t1 = "Aa";
        String t2 = "BB";
        System.out.println("s1 hashCode" + t1.hashCode());
        System.out.println("s1 hashCode" + t2.hashCode());
    }
}

结果

3、为什么要一起重写?

接下来回到本文的主题,重写 equals 为什么一定要重写 hashCode?​

为了解释这个问题,我们需要从下面的这个例子入手。

3.1 Set 正常使用

Set 集合是用来保存不同对象的,相同的对象就会被 Set 合并,最终留下一份独一无二的数据。​

import java.util.HashSet;
import java.util.Set;

public class HashCodeExample {
    public static void main(String[] args) {
        Set<String> set = new HashSet();
        set.add("Java");
        set.add("Java");
        set.add("MySQL");
        set.add("MySQL");
        set.add("Redis");
        System.out.println("Set 集合长度:" + set.size());
        System.out.println();
        // 打印 Set 中的所有元素
        set.forEach(d -> System.out.println(d));
    }
}

以上程序的执行结果,如下图所示

 从上述结果可以看出,重复的数据已经被 Set 集合“合并”了,这也是 Set 集合最大的特点:去重。

 3.2 Set 集合的“异常”

 只重写equals方法

package cn.itcast.interview;

import java.util.HashSet;
import java.util.Objects;
import java.util.Set;

/**
 * @author: msf
 * @date: 2022/11/26
 */
public class EqualsMyClassExample {
    public static void main(String[] args) {
        Person u1 = new Person();
        u1.setName("Java");
        u1.setAge(18);

        Person u2 = new Person();
        u2.setName("Java");
        u2.setAge(18);

        System.out.println("equals result = " + u1.equals(u2));

        // 创建 Set 集合
        Set<Person> set = new HashSet<Person>();
        set.add(u1);
        set.add(u2);
        // 打印 Set 中的所有数据
        set.forEach(System.out::println);
    }
}

class Person{
    private String name;
    private int age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = 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() {
        int result = name != null ? name.hashCode() : 0;
        result = 31 * result + age;
        return result;
    }*/

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

以上程序的执行结果,如下图所示

 这就造成了命名是相同值,为什么Set却重复存储呢?

下面对hashCode进行重写后结果

4 原因分析

默认情况下,Set 进行去重操作时,会先判断两个对象的 hashCode 是否相同,此时因为没有重写 hashCode 方法,所以会直接执行 Object 中的 hashCode 方法而 Object 中的 hashCode 方法对比的是两个不同引用地址的对象,所以结果是 false,那么 equals 方法就不用执行了,直接返回的结果就是 false:两个对象不是相等的,于是就在 Set 集合中插入了两个相同的对象。​

但是,如果在重写 equals 方法时,也重写了 hashCode 方法,那么在执行判断时会去执行重写的 hashCode 方法,此时对比的是两个对象的所有属性的 hashCode 是否相同,于是调用 hashCode 返回的结果就是 true,再去调用 equals 方法,发现两个对象确实是相等的,于是就返回 true 了,因此 Set 集合就不会存储两个一模一样的数据了,于是整个程序的执行就正常了。

总结

  • hashCode 和 equals 两个方法是用来协同判断两个对象是否相等的,采用这种方式的原因是可以提高程序插入和查询的速度。
  • 如果只重写equals方法,不重写hashCode方法,就有可能导致a.equals(b)这个表达式成立,但是hashCode却不同。会造成一个完全相同的对象会存储在hash表的不同位置。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/38750.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

第五届传智杯-初赛【A组-题解】

B题&#xff1a; 题目背景 【 题目背景和题目描述的两个题面是完全等价的&#xff0c;您可以选择阅读其中一部分。】 专攻超统一物理学的莲子&#xff0c;对机械结构的运动颇有了解。如下图所示&#xff0c;是一个三进制加法计算器的&#xff08;超简化&#xff09;示意图。…

idea,web开发中jsp页面中不提示控制层的请求地址

随着开发的进行&#xff0c;打开spring配置文件会有如下提示 同时工程管理里如下 删掉后&#xff0c;发现打开sping配置文件不告警了&#xff0c;可是jsp页面中也没有了地址的提示 这个提示没有了 正确的做法是删掉Spring Application Context 因为其他配置文件都导入App…

Java_接口

目录 1.接口的语法规则 2.接口使用 3.接口特性 4.实现多个接口 1&#xff09;下面通过类来表示一组动物&#xff1b; 2&#xff09;另外再提供一组接口, 分别表示 "会跑的", "会飞的", "会游泳的"&#xff1b; 3&#xff09;接下来我们创建…

黑马点评--达人探店

查看探店笔记&#xff1a; private void queryBlogUser(Blog blog) {Long userId blog.getUserId();User user userService.getById(userId);blog.setName(user.getNickName());blog.setIcon(user.getIcon()); }Override public Result queryBlogById(Long id) {//1.查询blo…

首1标准型和尾1标准型

目录 &#xff08;1&#xff09;系统的传递函数&#xff1b; (2) 系统的增益&#xff1b; &#xff08;3) 系统的特征根及相应的模态&#xff1b; &#xff08;4&#xff09; 画出对应的零极点图&#xff1b; &#xff08;5&#xff09; 求系统的单位脉冲响应&#…

Linux下JAVA使用JNA调用C++的动态链接库(g++或者gcc编译的.so文件)

目录 前言 一、准备工作 二、JAVA项目加载JNA 三、JNA的使用 3.1 生成.so文件 3.1.1 gcc生成的.so 3.1.2 g生成的.so 3.2 JNA调用.so 四、JAVA与C的类型对应 五、总结 前言 在没上班之前&#xff0c;我曾在CSDN写过《程序员的自我修养》的读书…

Centos7下载安装Tomcat应用服务器

Centos7安装Apache-Tomcat-9 文章目录Centos7安装Apache-Tomcat-91. 前言2. 官网下载Apache-Tomcat3. 使用xftp工具或者rz工具传输文件4. 解压Tomcat文件并启动Tomcat文件5. 关闭防火墙并访问Apache-Tomcat6. 可以修改端口号7. 开放端口1. 前言 Apache-Tomcat服务器是一个免费的…

Android 开发人脸识别之自动识别验证码功能讲解及实现(超详细 附源码)

需要源码和图片集请点赞关注收藏后评论区留下QQ或者私信~~~ 一、自动识别验证码 验证码图片中最简单的是数字验证码&#xff0c;一张再普通不过的验证码拿到之后要进行以下步骤的处理 1&#xff1a;首先对图片适当裁剪&#xff0c;先去掉外部的空白区域&#xff0c;再把每个数…

Oracle日志复制—国产自研Beedup(基于日志结构化数据复制产品)

Beedup能够实现大量交易数据的实时捕捉、变换和投递&#xff0c;实现源数据库与目标数据库的数据同步&#xff0c;保持亚秒级的数据延迟。 一、Beedup产品概述 由北京灵蜂纵横软件有限公司研发的数据库实时复制软件Beedup&#xff0c;提供数据库Oracle异地容灾备份&#xff0…

Vue笔记_01双向数据绑定原理

[1]什么叫双向数据绑定&#xff1f; 视图中的数据发生了变化&#xff0c;data中的数据也要对应改变&#xff1b;data中的数据发生了变化&#xff0c;视图上的数据也要对应改变&#xff1b; [2]双向绑定原理 vue 双向数据绑定是通过 数据劫持 结合 发布订阅模式的方式来实现的…

Pip版本问题导致Python模块安装失败

文章目录起因解决方案前言输入 %APPDATA%创建 pip.ini终端执行命令安装包结语起因 今天在视频号平台看到有小姐姐直播讲爬虫技术&#xff0c;我一看这不是挺简单的吗&#xff1f;于是我就想到自己四年前的今天刚进大一的时候最开始学的爬虫技术&#xff0c;后来因为一些事情跑…

怎么证明前端数据加密的三种方式

导读&#xff1a;前端最常见的三大加密方式https&#xff0c;SSH和MD5&#xff0c;这篇文章带你走进三大加密方式的原理对比。 1.https 1.1原理 A.就是在http加入SSL层&#xff0c;是http安全的基础; B.htts协议是在http基础上加了SSL协议; C.使用443端口&#xff0c;http是80…

2022CTF培训(三)windowslinux安卓平台调试机制原理

附件下载链接 windows平台调试机制原理 手动编写一个简易调试器 创建待调试进程 使用 CreateProcess 函数创建待调试进程&#xff0c;创建时指定 dwCreationFlags 参数为 DEBUG_ONLY_THIS_PROCESS 将会告诉操作系统我们需要让当前调用者&#xff08;线程&#xff09;接管所…

(十)centos7案例实战——实现nginx代理访问redis服务

前言 本节内容是关于实现nginx代理访问redis服务&#xff0c;由于在实际生产开发环境中&#xff0c;我们并不想将我们的中间键服务暴露在公网环境中&#xff0c;或者只能在内网环境中使用&#xff0c;例如本节内容&#xff0c;我们将redis安装到本地环境&#xff0c;但是又有需…

链表中倒数第k个结点、反转链表、合并两个排序的链表、树的子结构、删除链表中重复的结点

文章目录1、链表中倒数第k个结点2、反转链表3、合并两个排序的链表4、树的子结构5、 二叉树的镜像6、删除链表中重复的结点1、链表中倒数第k个结点 本题考点&#xff1a; 链表&#xff0c;前后指针的使用&#xff0c;边界条件检测 牛客链接 题目描述&#xff1a; 输入一个链表…

JVM【八股文】

JVM【八股文】 JVM内存区域划分 程序计数器栈堆方法区 一块大的区域&#xff0c;需要根据功能&#xff0c;来划分不同的小区域。 JVM内存是从操作系统里申请来的&#xff0c;之后堆这部分区域进行了划分。 1.程序计数器 内存中最小的区域&#xff0c;保存了下一条要执行指令…

android-加壳加固

title: android-加壳加固 categories: Android tags: [android, 加壳, 加固, 混淆] date: 2022-06-20 18:00:23 comments: false mathjax: true toc: true android-加壳 前篇 Android之Apk加壳 - https://blog.csdn.net/LVXIANGAN/article/details/84956476Android动态加载Dex…

李沐论文精度系列之七:Two-Stream双流网络、I3D

文章目录一、双流网络1.1 前言1.2 网络结构1.3 光流(Optical flow)1.3.1 什么是光流1.3.2 如何利用光流1.3.3 双向光流&#xff08;Bi-directional optical flow&#xff09;1.3.4 光流的局限性及和对应的预处理&#xff08;抽取&#xff09;方式1.3.5 视频模型测试1.4 实验1.4…

✿✿✿JavaScript基本语法一

目 录 1.js的发展史&#xff08;闲聊版&#xff09; 2.浏览器分成两部分&#xff1a;渲染引擎和 JS 引擎 3.js与html的关系以及结合方式 (1)js与html的关系 (2)js与html结合方式 4.JavaScript注释 5.js中的基本数据类型 6.js中的变量 7.运算符&#xff08;自动类型转…

9.前端笔记-CSS-盒子模型-border和padding

页面布局的三大核心&#xff1a; 盒子模型浮动定位 1、盒子模型 1.1 盒子模型组成 盒子模型本质还是一个盒子&#xff0c;包括边框border、外边距margin、内边距padding和实际内容content 1.1.1 边框border 组成 组成&#xff1a;颜色border-color、边框宽度border-wid…