持续总结中!2024年面试必问 100 道 Java基础面试题(三十九)

2024-05-11 02:04

本文主要是介绍持续总结中!2024年面试必问 100 道 Java基础面试题(三十九),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

上一篇地址:持续总结中!2024年面试必问 100 道 Java基础面试题(三十八)-CSDN博客

七十七、为什么重写equals就要重写hashCode方法?

在Java中,当重写equals()方法时,也应该重写hashCode()方法,这是因为hashCode()方法和equals()方法之间有一个重要的约定,这个约定对于所有依赖于hashCode()方法的集合类(如HashMapHashSetHashtable)的完整性至关重要。以下是详细解释为什么需要同时重写这两个方法:

约定

  1. 等价性:如果两个对象通过equals()方法比较是相等的,那么它们的hashCode()方法必须返回相同的值。

  2. 非等价性:如果两个对象通过equals()方法比较不相等,那么它们的hashCode()方法可以返回不同或相同的值。但是,为了提高哈希表的性能,建议在对象不相等时产生不同的哈希码。

原因

  1. 集合行为:很多基于哈希表的集合(如HashMapHashSet)依赖于对象的hashCode()值来确定对象的存储位置。如果两个对象相等(即equals()返回true),但它们的hashCode()值不同,这将导致这些集合无法正确地存储和检索对象。

  2. 性能:一致的hashCode()实现可以确保相等的对象被存储在哈希表的相同位置,这样可以提高查找、插入和删除操作的性能。

  3. 正确性:如果违反了equals()hashCode()之间的约定,那么基于哈希表的集合可能无法正确地工作。例如,HashMap可能无法根据键值检索到对象,或者HashSet可能会错误地认为集合中已经包含了一个相等的对象。

示例代码

import java.util.Objects;public class MyObject {private int id;private String name;public MyObject(int id, String name) {this.id = id;this.name = name;}@Overridepublic boolean equals(Object o) {if (this == o) return true;if (o == null || getClass() != o.getClass()) return false;MyObject myObject = (MyObject) o;return id == myObject.id && Objects.equals(name, myObject.name);}@Overridepublic int hashCode() {return Objects.hash(id, name);}
}

在这个示例中,MyObject类重写了equals()方法和hashCode()方法。equals()方法比较对象的idname属性,而hashCode()方法根据这些属性生成哈希码。

注意事项

  • 仅仅重写hashCode()方法而忽略equals()方法是不正确的,这可能导致哈希表操作出现问题。
  • 哈希码的计算应该基于对象的关键属性,这些属性是用于equals()方法比较的属性。
  • 在并发环境中,如果对象的属性可能会变化,那么hashCode()的值也应该相应地变化,以避免哈希表中的键失效。

总结来说,当重写equals()方法时,重写hashCode()方法是为了保持equals()hashCode()之间的约定,这对于确保使用哈希表的集合类的正确性和性能至关重要。

七十八、Java常用的元注解有哪些?

在Java中,元注解(Meta-Annotation)是指那些用来注解其他注解的注解。Java 5引入了元注解,它们提供了一种机制,允许开发者定义如何使用自定义注解。以下是Java中几种常用的元注解:

  1. @Target

    • 指定了注解所适用的元素类型,如类、方法、参数等。@Target本身有一个名为value的属性,它是一个注解类型数组,用来指定注解可以标记的元素类型。
    @Target(ElementType.METHOD)
    public @interface MyAnnotation {// 注解定义
    }
  2. @Retention

    • 定义了注解的保留策略,即注解在何时生效。@Retentionvalue属性可以是以下三个枚举值之一:
      • RetentionPolicy.SOURCE:注解仅在源代码中存在,编译后丢弃。
      • RetentionPolicy.CLASS:注解在编译成.class文件时保留,但JVM加载类时不会保留。
      • RetentionPolicy.RUNTIME:注解在运行时保留,可以通过反射读取。
    @Retention(RetentionPolicy.RUNTIME)
    public @interface MyAnnotation {// 注解定义
    }
  3. @Documented

    • 这个元注解没有任何属性,它用来标记注解是否应该被包含在用户文档中。通常与javadoc工具一起使用,如果一个注解被@Documented标记,那么使用该注解的程序元素也会在生成的javadoc中显示该注解。
    @Documented
    public @interface MyAnnotation {// 注解定义
    }
  4. @Inherited

    • 这个元注解指定了一个注解是否应该被子类继承。如果一个注解被@Inherited标记,那么这个注解在父类中的使用会被子类继承。
    @Inherited
    public @interface MyAnnotation {// 注解定义
    }
  5. @Repeatable

    • Java 8引入了这个元注解,它允许同一个注解在一个元素上使用多次。@Repeatablevalue属性指定了注解的重复使用是通过哪个容器注解来聚合的。
    @Repeatable(Annotations.class)
    public @interface SingleAnnotation {// 单个注解定义
    }@Retention(RetentionPolicy.RUNTIME)
    @Documented
    public @interface Annotations {SingleAnnotation[] value();
    }

这些元注解为注解的使用提供了额外的上下文信息,它们帮助定义注解的行为和限制,使得注解能够以预期的方式工作。正确使用元注解对于创建有效的、可维护的注解非常重要。

这篇关于持续总结中!2024年面试必问 100 道 Java基础面试题(三十九)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

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

字节面试 | 如何测试RocketMQ、RocketMQ?

字节面试:RocketMQ是怎么测试的呢? 答: 首先保证消息的消费正确、设计逆向用例,在验证消息内容为空等情况时的消费正确性; 推送大批量MQ,通过Admin控制台查看MQ消费的情况,是否出现消费假死、TPS是否正常等等问题。(上述都是临场发挥,但是RocketMQ真正的测试点,还真的需要探讨) 01 先了解RocketMQ 作为测试也是要简单了解RocketMQ。简单来说,就是一个分