java 如何重写equal 和hashcode方法(最佳实践)
先看完理解这篇:Java hashCode() 和 equals()的若干问题解答
实现高质量的equals方法的诀窍包括
- 使用==操作符检查“参数是否为这个对象的引用”;
- 使用instanceof操作符检查“参数是否为正确的类型”;
- 对于类中的关键属性,检查参数传入对象的属性是否与之相匹配;
- 编写完equals方法后,问自己它是否满足对称性、传递性、一致性;
- 重写equals时总是要重写hashCode;
- 不要将equals方法参数中的Object对象替换为其他的类型,在重写时不要忘掉@Override注解。
- public boolean equals(Object otherObject){ //测试两个对象是否是同一个对象,是的话返回true
- if(this == otherObject) { //测试检测的对象是否为空,是就返回false
- return true;
- }
- if(otherObject == null) { //测试两个对象所属的类是否相同,否则返回false
- return false;
- }
- if(getClass() != otherObject.getClass()) { //对otherObject进行类型转换以便和类A的对象进行比较
- return false;
- }
- A other=(A)otherObject;
- return Object.equals(类A对象的属性A,other的属性A)&&类A对象的属性B==other的属性B……;
- }
例子:
- public class TestEquals {
- public static void main(String[] args) {
- Person2 p1 = new Person2("aa", 13);
- Person2 p2 = new Person2("aa", 13);
- Person2 p3 = new Person2("bb", 13);
- System.out.println(p1.equals(p2)); // true
- System.out.println(p1.equals(p3)); // false
- }
- }
- class Person2 {
- private String name;
- private int age;
- public Person2(String name, int age) {
- this.name = name;
- this.age = age;
- }
- public String getName() {
- return name;
- }
- public void setName(String name) {
- this.name = name;
- }
- public int getAge() {
- return age;
- }
- public void setAge(int age) {
- this.age = age;
- }
- public boolean equals(Object another) {
- //先判断是不是自己,提高运行效率
- if (this == another)
- return true;
- //再判断是不是Person类,提高代码的健壮性
- if (another instanceof Person2) {
- //向下转型,父类无法调用子类的成员和方法
- Person2 anotherPerson = (Person2) another;
- //最后判断类的所有属性是否相等,其中String类型和Object类型可以用相应的equals()来判断
- if ((this.getName().equals(anotherPerson.getName())) && (this.getAge() == anotherPerson.getAge()))
- return true;
- } else {
- return false;
- }
- return false;
- }
- }
实现hashCode
方法的通用约定
在应用程序的执行期间,只要对象的
equals
方法的比较操作所用到的信息没有被修改,那么对这个同一对象调用多次,hashCode
方法必须始终如一地返回同一个整数。在同一个应用程序的多次执行过程中,每次执行所返回的整数可以不一致。如果两个对象根据
equals(Object)
方法比较是相等的,那么调用这两个对象中任意一个对象的hashCode
方法都必须产生同样的整数结果。反之,如果两个对象hashCode
方法返回整数结果一样,则不代表两个对象相等,因为equals
方法可以被重载。如果两个对象根据
equals(Object)
方法比较是不相等的,那么调用这两个对象中任意一个对象的hashCode
方法,则不一定要产生不同的整数结果。但,如果能让不同的对象产生不同的整数结果,则有可能提高散列表的性能。
hashCode
散列码计算(来自:Effective Java)
把某个非零的常数值,比如
17
,保存在一个名为result
的int
类型的变量中。对于对象中每个关键域
f
(指equals
方法中涉及的每个域),完成以下步骤:为该域计算
int
类型的散列码c:如果该域是
boolean
类型,则计算(f?1:0
)。如果该域是
byte
,char
,short
或者int类型,则计算(int)f
。如果该域是
long
类型,则计算(int)(f^(f>>>32))
。如果该域是
float
类型,则计算Float.floatToIntBits(f)
。如果该域是
double
类型,则计算Double.doubleToLongBits(f)
,然后按照步骤2.1.3,为得到的long
类型值计算散列值。如果该域是一个对象引用,并且该类的
equals
方法通过递归地调用equals
的方式来比较这个域,则同样为这个域递归地调用hashCode
。如果需要更复杂的比较,则为这个域计算一个范式(canonical representation)
,然后针对这个范式调用hashCode
。如果这个域的值为null
,则返回0
(其他常数也行)。如果该域是一个数组,则要把每一个元素当做单独的域来处理。也就是说,递归地应用上述规则,对每个重要的元素计算一个散列码,然后根据步骤2.2中的做法把这些散列值组合起来。如果数组域中的每个元素都很重要,可以利用发行版本1.5中增加的其中一个
Arrays.hashCode
方法。
按照下面的公式,把步骤2.1中计算得到的散列码
c
合并到result
中:result = 31 * result + c
; //此处31
是个奇素数,并且有个很好的特性,即用移位和减法来代替乘法,可以得到更好的性能:`31*i == (i<<5) - i, 现代JVM能自动完成此优化。
返回
result
检验并测试该
hashCode
实现是否符合通用约定。
- @Override
- public int hashCode() {
- int result = 17;
- result = 31 * result + mInt;
- result = 31 * result + (mBoolean ? 1 : 0);
- result = 31 * result + Float.floatToIntBits(mFloat);
- result = 31 * result + (int)(mLong ^ (mLong >>> 32));
- long mDoubleTemp = Double.doubleToLongBits(mDouble);
- result =31 * result + (int)(mDoubleTemp ^ (mDoubleTemp >>> 32));
- result = 31 * result + (mString == null ? 0 : mMsgContain.hashCode());
- result = 31 * result + (mObj == null ? 0 : mObj.hashCode());
- return result;
- }
java 如何重写equal 和hashcode方法(最佳实践)的更多相关文章
- Java中==、equals、hashcode的区别与重写equals以及hashcode方法实例(转)
Java中==.equals.hashcode的区别与重写equals以及hashcode方法实例 原文地址:http://www.cnblogs.com/luankun0214/p/4421770 ...
- java中equals方法和hashcode方法的区别和联系,以及为什么要重写这两个方法,不重写会怎样
一.在Object类中的定义为:public native int hashCode();是一个本地方法,返回的对象的地址值.但是,同样的思路,在String等封装类中对此方法进行了重写.方法调用得到 ...
- Java提高篇——equals()与hashCode()方法详解
java.lang.Object类中有两个非常重要的方法: 1 2 public boolean equals(Object obj) public int hashCode() Object类是类继 ...
- java集合(3)- Java中的equals和hashCode方法详解
参考:http://blog.csdn.net/jiangwei0910410003/article/details/22739953 Java中的equals方法和hashCode方法是Object ...
- Java中的equals和hashCode方法详解
Java中的equals和hashCode方法详解 转自 https://www.cnblogs.com/crazylqy/category/655181.html 参考:http://blog.c ...
- Java中的equals和hashCode方法
本文转载自:Java中的equals和hashCode方法详解 Java中的equals方法和hashCode方法是Object中的,所以每个对象都是有这两个方法的,有时候我们需要实现特定需求,可能要 ...
- 为什么要重写equals和hashcode方法
equals hashcode 当新建一个java类时,需要重写equals和hashcode方法,大家都知道!但是,为什么要重写呢? 需要保证对象调用equals方法为true时,hashcode ...
- 如何正确的重写equals() 和 hashCode()方法
比较两个Java对象时, 我们需要覆盖equals和 hashCode. public class User{ private String name; private int age; priva ...
- 转:Java中的equals和hashCode方法详解
转自:Java中的equals和hashCode方法详解 Java中的equals方法和hashCode方法是Object中的,所以每个对象都是有这两个方法的,有时候我们需要实现特定需求,可能要重写这 ...
随机推荐
- kvm:双网卡做bond+桥接
一,KVM基础 kvm是一种技术,云计算是一种模式,虚拟化是利用相应的技术方法在一台物理机器上将其按照不同的需求划分成多个相同或者不同的虚拟操作系统,并且各个虚拟系统可以同时运行,互不干扰,其中任何一 ...
- teradata安装
一,下载 步骤1 - 从链接下载所需的VM版本,http://downloads.teradata.com/download/database/teradata-express-for-vmware- ...
- CSS水印“点击穿透”
- luogu P1063 能量项链 x
P1063 能量项链 题目描述 在Mars星球上,每个Mars人都随身佩带着一串能量项链.在项链上有N颗能量珠.能量珠是一颗有头标记与尾标记的珠子,这些标记对应着某个正整数.并且,对于相邻的两颗珠子, ...
- 题解 CF1190B 【Tokitsukaze, CSL and Stone Game】
思路: 首先题目告诉我们,一次只能删去一个石子.当然有翻译时会注意,但是看英文题时总是容易忽略.. 先排序. 然后,你会发现,有些情况是一开始就输的,具体情况如下: 有两个 两个相等非零数.(a[x] ...
- Vim 命令、操作、快捷键(收藏大全)
------ 命令历史 以:和/开头的命令都有历史纪录,可以首先键入:或/然后按上下箭头来选择某个历史命令. 启动vim 在命令行窗口中输入以下命令即可 vim 直接启动vim vim filenam ...
- 7.Python编码规范(PEP 8)
在讲解具体的 Python 编码规范之前,先来看看图 1 中的代码: 图 1 两段功能相同的 Python 代码 对比图 1 中的两段代码你会发现,它们所包含的代码时完全相同的,但很明显,右侧的代码编 ...
- Linux安装部署FTP服务器
Linux安装部署FTP服务器 本文章会将安装FTP服务器的步骤以及一些遇到的问题来记录下 因为项目中要与第三方对接数据,需要用到FTP服务器以提供他们每天上传数据,因为之前在本地的VMware虚 ...
- 当出现no changes added to commit时如何正确使用git提交命令
对于这个问题,最好的解决方法就是按如下步骤:1.到根目录下:git add . :("."是必须要的)2.git commit -m "some word"3 ...
- php 防盗链
防盗链的技术已经很普遍了,有些网站不喜欢自己的图片被别的网站直接复制使用,便使用了防盗链的技术,这样别人在直接复制使用网站图片时,图片便会按照程序的设定不显示或显示防盗链等字样. 使用了防盗链技术,不 ...