整理好了!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

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

2024年流动式起重机司机证模拟考试题库及流动式起重机司机理论考试试题

题库来源:安全生产模拟考试一点通公众号小程序 2024年流动式起重机司机证模拟考试题库及流动式起重机司机理论考试试题是由安全生产模拟考试一点通提供,流动式起重机司机证模拟考试题库是根据流动式起重机司机最新版教材,流动式起重机司机大纲整理而成(含2024年流动式起重机司机证模拟考试题库及流动式起重机司机理论考试试题参考答案和部分工种参考解析),掌握本资料和学校方法,考试容易。流动式起重机司机考试技

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟 开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚 第一站:海量资源,应有尽有 走进“智听