整理好了!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通过驱动包(jar包)连接MySQL数据库的步骤总结及验证方式

《Java通过驱动包(jar包)连接MySQL数据库的步骤总结及验证方式》本文详细介绍如何使用Java通过JDBC连接MySQL数据库,包括下载驱动、配置Eclipse环境、检测数据库连接等关键步骤,... 目录一、下载驱动包二、放jar包三、检测数据库连接JavaJava 如何使用 JDBC 连接 mys

SpringBoot线程池配置使用示例详解

《SpringBoot线程池配置使用示例详解》SpringBoot集成@Async注解,支持线程池参数配置(核心数、队列容量、拒绝策略等)及生命周期管理,结合监控与任务装饰器,提升异步处理效率与系统... 目录一、核心特性二、添加依赖三、参数详解四、配置线程池五、应用实践代码说明拒绝策略(Rejected

SQL中如何添加数据(常见方法及示例)

《SQL中如何添加数据(常见方法及示例)》SQL全称为StructuredQueryLanguage,是一种用于管理关系数据库的标准编程语言,下面给大家介绍SQL中如何添加数据,感兴趣的朋友一起看看吧... 目录在mysql中,有多种方法可以添加数据。以下是一些常见的方法及其示例。1. 使用INSERT I

Python中反转字符串的常见方法小结

《Python中反转字符串的常见方法小结》在Python中,字符串对象没有内置的反转方法,然而,在实际开发中,我们经常会遇到需要反转字符串的场景,比如处理回文字符串、文本加密等,因此,掌握如何在Pyt... 目录python中反转字符串的方法技术背景实现步骤1. 使用切片2. 使用 reversed() 函

一文详解SpringBoot中控制器的动态注册与卸载

《一文详解SpringBoot中控制器的动态注册与卸载》在项目开发中,通过动态注册和卸载控制器功能,可以根据业务场景和项目需要实现功能的动态增加、删除,提高系统的灵活性和可扩展性,下面我们就来看看Sp... 目录项目结构1. 创建 Spring Boot 启动类2. 创建一个测试控制器3. 创建动态控制器注

Java操作Word文档的全面指南

《Java操作Word文档的全面指南》在Java开发中,操作Word文档是常见的业务需求,广泛应用于合同生成、报表输出、通知发布、法律文书生成、病历模板填写等场景,本文将全面介绍Java操作Word文... 目录简介段落页头与页脚页码表格图片批注文本框目录图表简介Word编程最重要的类是org.apach

Spring Boot中WebSocket常用使用方法详解

《SpringBoot中WebSocket常用使用方法详解》本文从WebSocket的基础概念出发,详细介绍了SpringBoot集成WebSocket的步骤,并重点讲解了常用的使用方法,包括简单消... 目录一、WebSocket基础概念1.1 什么是WebSocket1.2 WebSocket与HTTP

SpringBoot+Docker+Graylog 如何让错误自动报警

《SpringBoot+Docker+Graylog如何让错误自动报警》SpringBoot默认使用SLF4J与Logback,支持多日志级别和配置方式,可输出到控制台、文件及远程服务器,集成ELK... 目录01 Spring Boot 默认日志框架解析02 Spring Boot 日志级别详解03 Sp

java中反射Reflection的4个作用详解

《java中反射Reflection的4个作用详解》反射Reflection是Java等编程语言中的一个重要特性,它允许程序在运行时进行自我检查和对内部成员(如字段、方法、类等)的操作,本文将详细介绍... 目录作用1、在运行时判断任意一个对象所属的类作用2、在运行时构造任意一个类的对象作用3、在运行时判断

java如何解压zip压缩包

《java如何解压zip压缩包》:本文主要介绍java如何解压zip压缩包问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java解压zip压缩包实例代码结果如下总结java解压zip压缩包坐在旁边的小伙伴问我怎么用 java 将服务器上的压缩文件解压出来,