本文主要是介绍持续总结中!2024年面试必问 100 道 Java基础面试题(三十九),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
上一篇地址:持续总结中!2024年面试必问 100 道 Java基础面试题(三十八)-CSDN博客
七十七、为什么重写equals就要重写hashCode方法?
在Java中,当重写equals()
方法时,也应该重写hashCode()
方法,这是因为hashCode()
方法和equals()
方法之间有一个重要的约定,这个约定对于所有依赖于hashCode()
方法的集合类(如HashMap
、HashSet
和Hashtable
)的完整性至关重要。以下是详细解释为什么需要同时重写这两个方法:
约定
-
等价性:如果两个对象通过
equals()
方法比较是相等的,那么它们的hashCode()
方法必须返回相同的值。 -
非等价性:如果两个对象通过
equals()
方法比较不相等,那么它们的hashCode()
方法可以返回不同或相同的值。但是,为了提高哈希表的性能,建议在对象不相等时产生不同的哈希码。
原因
-
集合行为:很多基于哈希表的集合(如
HashMap
、HashSet
)依赖于对象的hashCode()
值来确定对象的存储位置。如果两个对象相等(即equals()
返回true
),但它们的hashCode()
值不同,这将导致这些集合无法正确地存储和检索对象。 -
性能:一致的
hashCode()
实现可以确保相等的对象被存储在哈希表的相同位置,这样可以提高查找、插入和删除操作的性能。 -
正确性:如果违反了
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()
方法比较对象的id
和name
属性,而hashCode()
方法根据这些属性生成哈希码。
注意事项
- 仅仅重写
hashCode()
方法而忽略equals()
方法是不正确的,这可能导致哈希表操作出现问题。 - 哈希码的计算应该基于对象的关键属性,这些属性是用于
equals()
方法比较的属性。 - 在并发环境中,如果对象的属性可能会变化,那么
hashCode()
的值也应该相应地变化,以避免哈希表中的键失效。
总结来说,当重写equals()
方法时,重写hashCode()
方法是为了保持equals()
和hashCode()
之间的约定,这对于确保使用哈希表的集合类的正确性和性能至关重要。
七十八、Java常用的元注解有哪些?
在Java中,元注解(Meta-Annotation)是指那些用来注解其他注解的注解。Java 5引入了元注解,它们提供了一种机制,允许开发者定义如何使用自定义注解。以下是Java中几种常用的元注解:
-
@Target
:- 指定了注解所适用的元素类型,如类、方法、参数等。
@Target
本身有一个名为value
的属性,它是一个注解类型数组,用来指定注解可以标记的元素类型。
@Target(ElementType.METHOD) public @interface MyAnnotation {// 注解定义 }
- 指定了注解所适用的元素类型,如类、方法、参数等。
-
@Retention
:- 定义了注解的保留策略,即注解在何时生效。
@Retention
的value
属性可以是以下三个枚举值之一:RetentionPolicy.SOURCE
:注解仅在源代码中存在,编译后丢弃。RetentionPolicy.CLASS
:注解在编译成.class
文件时保留,但JVM加载类时不会保留。RetentionPolicy.RUNTIME
:注解在运行时保留,可以通过反射读取。
@Retention(RetentionPolicy.RUNTIME) public @interface MyAnnotation {// 注解定义 }
- 定义了注解的保留策略,即注解在何时生效。
-
@Documented
:- 这个元注解没有任何属性,它用来标记注解是否应该被包含在用户文档中。通常与
javadoc
工具一起使用,如果一个注解被@Documented
标记,那么使用该注解的程序元素也会在生成的javadoc中显示该注解。
@Documented public @interface MyAnnotation {// 注解定义 }
- 这个元注解没有任何属性,它用来标记注解是否应该被包含在用户文档中。通常与
-
@Inherited
:- 这个元注解指定了一个注解是否应该被子类继承。如果一个注解被
@Inherited
标记,那么这个注解在父类中的使用会被子类继承。
@Inherited public @interface MyAnnotation {// 注解定义 }
- 这个元注解指定了一个注解是否应该被子类继承。如果一个注解被
-
@Repeatable
:- Java 8引入了这个元注解,它允许同一个注解在一个元素上使用多次。
@Repeatable
的value
属性指定了注解的重复使用是通过哪个容器注解来聚合的。
@Repeatable(Annotations.class) public @interface SingleAnnotation {// 单个注解定义 }@Retention(RetentionPolicy.RUNTIME) @Documented public @interface Annotations {SingleAnnotation[] value(); }
- Java 8引入了这个元注解,它允许同一个注解在一个元素上使用多次。
这些元注解为注解的使用提供了额外的上下文信息,它们帮助定义注解的行为和限制,使得注解能够以预期的方式工作。正确使用元注解对于创建有效的、可维护的注解非常重要。
这篇关于持续总结中!2024年面试必问 100 道 Java基础面试题(三十九)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!