正确重写equals和hashCode需遵守自反性、对称性、传递性、一致性和非空性;先判断引用是否相同,再通过instanceof检查类型并比较关键字段;若重写equals则必须重写hashCode,确保相等对象具有相同哈希码,通常用31乘以各关键字段哈希值累加,以提升哈希表性能。

在Java中正确重写 equals 和 hashCode 方法是对象比较和集合操作的基础。如果处理不当,可能导致逻辑错误,尤其是在使用 HashMap、HashSet 等基于哈希的集合时。
默认情况下,Object 类的 equals 方法比较的是对象的内存地址(即 == 比较)。当我们需要根据对象的实际内容判断相等性时,就必须重写该方法。
重写 equals 方法需遵守以下约定(来自 Object 类规范):
实际编写时建议遵循以下步骤:
立即学习“Java免费学习笔记(深入)”;
public class Person {
private String id;
private String name;
<pre class='brush:java;toolbar:false;'>@Override
public boolean equals(Object obj) {
if (this == obj) return true;
if (obj == null || getClass() != obj.getClass()) return false;
Person person = (Person) obj;
return id.equals(person.id) && name.equals(person.name);
}}
根据 Java 规范,如果两个对象通过 equals 判断相等,那么它们的 hashCode 必须相同。反之则不一定成立(即哈希码相同不代表对象相等)。
因此,只要重写了 equals,就必须重写 hashCode,否则会破坏哈希集合的行为。
设计原则:
常见实现方式是组合关键字段的哈希值:
// 示例:Person 类的 hashCode 实现
@Override
public int hashCode() {
int result = id.hashCode();
result = 31 * result + name.hashCode();
return result;
}
数字 31 是一个奇素数,编译器可优化为位运算(31 * i == (i
手动编写 equals 和 hashCode 容易出错,尤其是字段较多时。推荐使用工具辅助生成:
@EqualsAndHashCode
public class Person {
private String id;
private String name;
}
注意:Lombok 默认包含所有非静态字段。如需排除某些字段,可用 @EqualsAndHashCode(exclude = {"name"})。
如果对象的状态在创建后不再改变(即不可变对象),那么其 equals 和 hashCode 更加稳定,适合做 Map 的 key 或 Set 的元素。
建议:
这样可以防止对象放入 HashSet 后因字段变化导致无法查找的问题。
基本上就这些。只要记住:重写 equals 一定要重写 hashCode,字段决定相等性,哈希影响性能,工具能帮你少犯错。
以上就是如何在Java中正确重写 equals 和 hashCode_对象比较的设计要点的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号