整理好了!2024年最常见 100 道 Java基础面试题(四十二)

2024-05-13 03:36

本文主要是介绍整理好了!2024年最常见 100 道 Java基础面试题(四十二),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

上一篇地址:整理好了!2024年最常见 100 道 Java基础面试题(四十一)-CSDN博客

八十三、 两个对象 hashCode 相等,equals 也相等么?

在Java中,如果两个对象的 hashCode() 方法返回相同的散列码,这并不意味着这两个对象通过 equals() 方法比较也一定相等。hashCode()equals() 方法之间的关系是单向的:如果两个对象相等(即 equals() 返回 true),那么它们的散列码必须相同;但散列码相同并不意味着对象相等。

原因

  1. 散列码的碰撞(Collision):由于散列码的值域有限,而对象可能非常多,不同的对象可能会生成相同的散列码,这种现象称为“碰撞”。

  2. 散列码的分布:一个好的散列函数会尽量减少碰撞的发生,使得散列码尽可能均匀分布,但无法保证完全没有碰撞。

  3. 散列码的计算:散列码通常是根据对象的某些属性计算得到的,即使两个对象的这些属性值相同,也不代表它们在所有属性上都相同。

示例

public class Person {private String name;private int age;public Person(String name, int age) {this.name = name;this.age = age;}@Overridepublic boolean equals(Object obj) {if (this == obj) return true;if (obj == null || getClass() != obj.getClass()) return false;Person person = (Person) obj;return age == person.age && Objects.equals(name, person.name);}@Overridepublic int hashCode() {// 简化的hashCode计算方式,可能导致碰撞return Objects.hash(name, age);}
}

在这个示例中,即使两个 Person 对象具有不同的 name 但相同的 age,它们的 hashCode() 方法也可能返回相同的散列码,但通过 equals() 方法比较时,它们是不相等的。

注意事项

  • 在设计 hashCode() 方法时,应该尽量减少碰撞的发生,以提高散列表的性能。
  • 当使用散列表(如 HashMapHashSet)时,即使两个对象的散列码相同,散列表也通常能够通过链地址法、开放寻址法等方式处理碰撞,正确存储和检索对象。

总结

两个对象的 hashCode() 方法返回相同的散列码,并不意味着它们通过 equals() 方法比较相等。这是由于散列码的计算可能发生碰撞,且散列码的值域有限。在实际应用中,应该正确实现 equals()hashCode() 方法,以确保对象的相等性和散列表的性能。

八十四、为什么重写 equals 就要重写 hashCode 方法

在Java中,当你重写了 equals() 方法来定义对象的等价性时,也应该重写 hashCode() 方法,以维护Object类中equals()方法和hashCode()方法之间的约定。这个约定确保了对象的相等性和散列码的一致性,对于依赖于散列码的集合(如HashMapHashSet)的正确运作至关重要。以下是重写hashCode()方法的原因:

1. 一致性

当两个对象通过equals()方法比较结果为true时,它们的hashCode()方法必须返回相同的散列码。如果只重写了equals()方法而没有重写hashCode()方法,就会违反这个约定,导致相同对象的散列码不一致。

2. 散列表的性能

散列表的性能在很大程度上依赖于散列码的均匀分布。如果对象的equals()方法被重写以定义了一个新的相等性概念,而hashCode()方法仍然是默认的实现,那么可能会导致散列码分布不均匀,增加散列表的冲突概率,从而降低散列表的性能。

3. 依赖于散列码的集合

对于依赖于散列码的集合,如HashMapHashSet,对象的相等性检查和散列码计算通常是成对出现的。如果equals()hashCode()方法的一致性被破坏,可能会导致以下问题:

  • 对象无法通过HashMapget()方法正确检索。
  • 对象可能无法正确地添加到HashSet中或被识别为已存在。

4. 跨系统的兼容性

在分布式系统或序列化场景中,对象的散列码可能用于不同系统间的对象识别。如果equals()hashCode()方法的一致性被破坏,可能会导致跨系统间的对象识别失败。

示例

public class Person {private final String name;private final int age;public Person(String name, int age) {this.name = name;this.age = age;}@Overridepublic boolean equals(Object obj) {if (this == obj) return true;if (obj == null || getClass() != obj.getClass()) return false;Person person = (Person) obj;return age == person.age && Objects.equals(name, person.name);}@Overridepublic int hashCode() {return Objects.hash(name, age);}
}

在这个示例中,Person类的equals()方法比较了两个对象的nameage属性。为了保持一致性,hashCode()方法也被重写,以确保具有相同属性的对象有相同的散列码。

注意事项

  • 重写equals()方法时,必须同时重写hashCode()方法。
  • hashCode()方法的实现应该保证在Java应用程序执行期间的一致性,即使对象的属性值被修改,只要对象仍然被认为是相等的,它们的散列码就应该保持不变。

总结

重写equals()方法而不重写hashCode()方法会违反对象相等性和散列码一致性的约定,这可能会导致依赖于散列码的集合出现错误。为了保持一致性并确保散列表的性能,当你重写equals()方法时,也应该重写hashCode()方法。

这篇关于整理好了!2024年最常见 100 道 Java基础面试题(四十二)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/984570

相关文章

Java实现字节字符转bcd编码

《Java实现字节字符转bcd编码》BCD是一种将十进制数字编码为二进制的表示方式,常用于数字显示和存储,本文将介绍如何在Java中实现字节字符转BCD码的过程,需要的小伙伴可以了解下... 目录前言BCD码是什么Java实现字节转bcd编码方法补充总结前言BCD码(Binary-Coded Decima

SpringBoot全局域名替换的实现

《SpringBoot全局域名替换的实现》本文主要介绍了SpringBoot全局域名替换的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录 项目结构⚙️ 配置文件application.yml️ 配置类AppProperties.Ja

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

JavaScript中的高级调试方法全攻略指南

《JavaScript中的高级调试方法全攻略指南》什么是高级JavaScript调试技巧,它比console.log有何优势,如何使用断点调试定位问题,通过本文,我们将深入解答这些问题,带您从理论到实... 目录观点与案例结合观点1观点2观点3观点4观点5高级调试技巧详解实战案例断点调试:定位变量错误性能分

Java实现将HTML文件与字符串转换为图片

《Java实现将HTML文件与字符串转换为图片》在Java开发中,我们经常会遇到将HTML内容转换为图片的需求,本文小编就来和大家详细讲讲如何使用FreeSpire.DocforJava库来实现这一功... 目录前言核心实现:html 转图片完整代码场景 1:转换本地 HTML 文件为图片场景 2:转换 H

Java使用jar命令配置服务器端口的完整指南

《Java使用jar命令配置服务器端口的完整指南》本文将详细介绍如何使用java-jar命令启动应用,并重点讲解如何配置服务器端口,同时提供一个实用的Web工具来简化这一过程,希望对大家有所帮助... 目录1. Java Jar文件简介1.1 什么是Jar文件1.2 创建可执行Jar文件2. 使用java

SpringBoot实现不同接口指定上传文件大小的具体步骤

《SpringBoot实现不同接口指定上传文件大小的具体步骤》:本文主要介绍在SpringBoot中通过自定义注解、AOP拦截和配置文件实现不同接口上传文件大小限制的方法,强调需设置全局阈值远大于... 目录一  springboot实现不同接口指定文件大小1.1 思路说明1.2 工程启动说明二 具体实施2

Java实现在Word文档中添加文本水印和图片水印的操作指南

《Java实现在Word文档中添加文本水印和图片水印的操作指南》在当今数字时代,文档的自动化处理与安全防护变得尤为重要,无论是为了保护版权、推广品牌,还是为了在文档中加入特定的标识,为Word文档添加... 目录引言Spire.Doc for Java:高效Word文档处理的利器代码实战:使用Java为Wo

SpringBoot日志级别与日志分组详解

《SpringBoot日志级别与日志分组详解》文章介绍了日志级别(ALL至OFF)及其作用,说明SpringBoot默认日志级别为INFO,可通过application.properties调整全局或... 目录日志级别1、级别内容2、调整日志级别调整默认日志级别调整指定类的日志级别项目开发过程中,利用日志

Java中的抽象类与abstract 关键字使用详解

《Java中的抽象类与abstract关键字使用详解》:本文主要介绍Java中的抽象类与abstract关键字使用详解,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、抽象类的概念二、使用 abstract2.1 修饰类 => 抽象类2.2 修饰方法 => 抽象方法,没有