重写了equals()方法之后也必须重写hashCode()方法

我们都知道Java语言是完全面向对象的,在java中,所有的对象都是继承于Object类。Ojbect类中有两个方法equals、hashCode,这两个方法都是用来比较两个对象是否相等的。

在未重写equals方法我们是继承了object的equals方法,那里的 equals是比较两个对象的内存地址,显然我们new了2个对象内存地址肯定不一样

  • 对于值对象,==比较的是两个对象的值
  • 对于引用对象,比较的是两个对象的地址

默认的equals方法同==,一般来说我们的对象都是引用对象,要重写equals方法。

现在有一个学生对象,有属性学号跟姓名,现在我新建了一个学生对象,又从数据里查出一个学生对象,这两个对象的学号跟姓名都一样,那这两个对象是不是相等呢?一般情况下,除非你有特殊需求要处理,这两个对象是相等的,可如果用==去比较,返回的结果是错误的。

这时候我们就必须重写equlas方法了。如果学号是主键,在equals方法里,我们认为只要学号相同,就可以返回true。

hashCode方法也是可以用来比较两个对象是否相等的。但是我们很少使用,应该说是很少直接使用。hashCode方法返回的是一个int值,可以看做是一个对象的唯一编码,


hashCode主要为散列表(如HashMap)提供支持。
如果一个对象的equals没有被改写,那么在执行时无论调用多少次hashCode()总是返回相同的值(对象的内部地址转化为该数值)。
如果两个对象相等(equals相等),那么对应的hashCode也相等。
两个对象不相等(equals不相等),其对应的hashCode也不相等,这种规范不是必须的(亦即两个对象不相等,其对应的hashCode可能相等)。延伸理解,hashCode相等的两个对象不一定相等(equals相等)。但是建议开发者遵守为两个不同对象返回不同的散列码的规范(比如重写时,但是Object.hashCode()确实为不同的对象返回不同的hashCode), 这种可以提升hash tables的性能。
特殊情况:并不一定每次执行hashCode()方法都返回完全一样的值。(参考JavaDoc)尽管不常见,但是确实有些Java库实际上在不同执行的过程中返回不同的hashCode值,如 Google’s Protocol Buffers[size=medium][/size]

 

一般如果使用java中的Map对象进行存储时,他会自动调用hashCode方法来比较两个对象是否相等。

所以如果我们对equals方法进行了重写,建议一定要对hashCode方法重写,以保证相同的对象返回相同的hash值,不同的对象返回不同的hash值。

如上面的学生例子,如果学号相同,不管姓名相不相同,返回的hash值一定要是一样的,这时我们的hash值只与学号有关。

Java代码  重写了equals()方法之后也必须重写hashCode()方法
  1. public class Test    
  2.  {    
  3.      public static void main(String[] args)    
  4.      {    
  5.      HashMap hm = new HashMap();    
  6.          
  7.      hm.put(new key(1),new value(2));    
  8.      if(hm.containsKey(new key(1)))    
  9.       System.out.println(hm.get(new key(1)));    
  10.      else    
  11.       System.out.println("dont have such a key");    
  12.      }    
  13.  }    

 你每次new 一个新对象出来hashcode肯定不一样,所以你拿不到你要的key。

Java代码  重写了equals()方法之后也必须重写hashCode()方法
  1. class key    
  2.  {    
  3.      int i ;    
  4.      public key(int i)    
  5.      {    
  6.      this.i = i;    
  7.      }    
  8.      @Override    
  9.      public boolean equals(Object obj)    
  10.      {    
  11.      if(obj instanceof key)    
  12.      {    
  13.       if(((key)obj).i == i)    
  14.          return true;    
  15.      }    
  16.      return false;    
  17.      }    
  18.      @Override    
  19.      public int hashCode()    
  20.      {    
  21.      return i;    
  22.      }    
  23.  }    

 

1、重写equals方法时需要重写hashCode方法,主要是针对Map、Set等集合类型的使用;

a: Map、Set等集合类型存放的对象必须是唯一的;

b: 集合类判断两个对象是否相等,是先判断equals是否相等,如果equals返回TRUE,还要再判断HashCode返回值是否ture,只有两者都返回ture,才认为该两个对象是相等的。

2、由于Object的hashCode返回的是对象的hash值,所以即使equals返回TRUE,集合也可能判定两个对象不等,所以必须重写hashCode方法,以保证当equals返回TRUE时,hashCode也返回Ture,这样才能使得集合中存放的对象唯一。

c
Co
Cod
Code
上一篇:css3的高级而有用且很少人知道的属性和样式


下一篇:Apache Spark源码走读(八)Graphx实现剖析&spark repl实现详解