欢迎讨论、交流,转载请注明出处,3Q!

       前言

             一直对hashCode()这个方法由困惑,也不知道什么时候该去实现hashCode()方法。趁着研究

          源码的这段时间吧hashCode()方法做一个总结。

       HashCode返回值

              Java中每个对象都是Object的子类,也就用于hashCode()方法,查看源码可以知道,hashCode()

          方法返回的是一个整数,这有什么实际含义吗?

              其实hashCode()方法返回的是对象的哈希码,那么对象的哈希码又是个什么东东?

          简单的来书对象的哈希码代表了对象的特征。对象的特征有阵各种的表现形式,例如对象的物理

          地址,字符串内容,或者数字算出来的int类型的数值。哈希码既然用于表示对象的特征,就可以

          用于判断对象是否相等,所以在实际情况下如果对象通过equals方法是相等的,那么其hashCode

          返回的值也是相等的。例如String类。  

package com.kiritor;  public class HashTest {   public static void main(String[] args) { 	String a = "AAA"; 	String b ="AAA"; 	System.out.println(a.hashCode()); 	System.out.println(b.hashCode()); } } 
              两句Sysout语句输出的结果都是一样的!Object的hashCode()方法其实就是根据对象的物理地

          址生成的hash值,Object的equals方法判断的也是物理地址。这里由于Object的hashCode方法是

          native的我们就解析下String 类的hashCode的源码吧:

 public int hashCode() {         int h = hash;         if (h == 0 && value.length > 0) {             char val[] = value;              for (int i = 0; i < value.length; i++) {                 h = 31 * h + val[i];             }             hash = h;         }         return h;     } 
            可以看见的是String的hashCode就是根据字符串的内容来的,而其equals方法比较的也是字符串的

         内容。

   public boolean equals(Object anObject) {         if (this == anObject) {             return true;         }         if (anObject instanceof String) {             String anotherString = (String) anObject;             int n = value.length;             if (n == anotherString.value.length) {                 char v1[] = value;                 char v2[] = anotherString.value;                 int i = 0;                 while (n-- != 0) {                     if (v1[i] != v2[i])                             return false;                     i++;                 }                 return true;             }         }         return false;     } 

       哈希码的作用   

               现在我们知道了hashCode方法产生的是对象的哈希值(对象特征)。但是对于对象的比较我们有equals

           方法了,还需要hashCode()方法干什么?

               其一:在不自己实现hashCode()方法的情况下描述的是对象的物理地址信息,而我们一般重写equals

           方法进行对象的判等,对于对象的物理地址的描述信息有必要保存。

               其二:java集合中存在一种无序但元素不重复的集合例如HashMap,保证元素不重复这个可以通过

           equals方法来实现,不过当元素过多的时候,后添加的元素必然比较的次数过多。此时hashCode就非常

           有用了,因为其返回的是一个整数,我们可以对这个哈希值做相关处理形成对象插入数组的索引。

               实现内部哈希排序:哈希映射技术。

               关于内部哈希排序可参考这篇文章:

                http://blog.csdn.net/kiritor/article/details/8884371

                在HashMap集合映射中采用了哈希表的原理,哈希算法也成为散列算法,就是将数据依照特定的

          算法直接指定到一个地址上。

               当集合要添加元素的时候,首先得到这个元素的hashCode方法,在通过hash算法定位到放置的物理

          地址,若此刻这个位置没有元素则直接存储在这个位置,若有则调用equals判断相同就不存,不同的话就

          重新进行散列(这里并没有针对哪种集合只是说明这个问题)

         总结

                  hashCode()方法返回的是对象的特征(物理地址、内容等),他和equals方法比起来更像是对象的编码

           equals更像是对象的内容判断。

                  如果两个对象是相等的,那么他们的hashCode方法一定要相等才是合理的;若果两个对象的

           hashCode相等,他们可能是不相等的。上述对象的相等是有equals判断的。

                  因此对于equals方法和hashCode方法来说,他们总应该保持逻辑上的一致关系。在重写equals方法的

           时候也尽量的重写hashCode方法。