HashCode总结

来源:互联网 发布:控制中心添加蜂窝数据 编辑:程序博客网 时间:2024/06/05 12:00

**

什么是Hashcode?

**
简单的来讲HashCode就像是一个签名,当两个对象的Hashcode一样的时候,两个对象就可能一样,但如果Hashcode不一样,那么肯定不是同一个对象。相当于先确定一个大的范围,再用equals去比较。
一个对象的散列码,什么是散列码呢,简单的说就是通过哈希算法算出来的一大窜数字之类的东西和内存有关.

如果对象1和对象2相等,说明他们的散列码相等!反过来就不一样了!
hashcode可以减少equals比较的次数,提高运算效率。
一般用hashcode来进行比较两个东西是不是一样的,可以很容易的排除许多不一样的东西。

以上这段官方文档的定义,我们可以抽出成以下几个关键点
1、hashCode的存在主要是用于查找的快捷性,如Hashtable,HashMap等,hashCode是用来在散列存储结构中确定对象的存储地址的;
2、如果两个对象相同,就是适用于equals(Java.lang.Object) 方法,那么这两个对象的hashCode一定要相同;
3、如果对象的equals方法被重写,那么对象的hashCode也尽量重写,并且产生hashCode使用的对象,一定要和equals方法中使用的一致,否则就会违反上面提到的第2点;
4、两个对象的hashCode相同,并不一定表示两个对象就相同,也就是不一定适用于equals(java.lang.Object) 方法,只能够说明这两个对象在散列存储结构中,如Hashtable,他们“存放在同一个篮子里”。

以下的代码是从其他帖子拷贝过来的。可以参考一下:
1.hashcode是用来查找的,如果你学过数据结构就应该知道,在查找和排序这一章有
例如内存中有这样的位置
0 1 2 3 4 5 6 7
而我有个类,这个类有个字段叫ID,我要把这个类存放在以上8个位置之一,如果不用hashcode而任意存放,那么当查找时就需要到这八个位置里挨个去找,或者用二分法一类的算法。
但如果用hashcode那就会使效率提高很多。
我们这个类中有个字段叫ID,那么我们就定义我们的hashcode为ID%8,然后把我们的类存放在取得得余数那个位置。比如我们的ID为9,9除8的余数为1,那么我们就把该类存在1这个位置,如果ID是13,求得的余数是5,那么我们就把该类放在5这个位置。这样,以后在查找该类时就可以通过ID除 8求余数直接找到存放的位置了。

2.但是如果两个类有相同的hashcode怎么办那(我们假设上面的类的ID不是唯一的),例如9除以8和17除以8的余数都是1,那么这是不是合法的,回答是:可以这样。那么如何判断呢?在这个时候就需要定义 equals了。
也就是说,我们先通过 hashcode来判断两个类是否存放某个桶里,但这个桶里可能有很多类,那么我们就需要再通过 equals 来在这个桶里找到我们要的类。
那么。重写了equals(),为什么还要重写hashCode()呢?
想想,你要在一个桶里找东西,你必须先要找到这个桶啊,你不通过重写hashcode()来找到桶,光重写equals()有什么用啊 。
下面给出代码来看一看:

public class HashTest {      private int i;      public int getI() {          return i;      }      public void setI(int i) {          this.i = i;      }      public int hashCode() {          return i % 10;      }      public final static void main(String[] args) {          HashTest a = new HashTest();          HashTest b = new HashTest();          a.setI(1);          b.setI(1);          Set<HashTest> set = new HashSet<HashTest>();          set.add(a);          set.add(b);          System.out.println(a.hashCode() == b.hashCode());          System.out.println(a.equals(b));          System.out.println(set);      }  }  这样输出的结果是true  false  [com.ubs.sae.test.HashTest@1, com.ubs.sae.test.HashTest@1]   //之所以是FALSE是因为系统调用的equals()方法是Object中的equals()方法,比较的是两个对象的引用是否相同,这两个对象的引用显然是不相同的,所以返回false//下面我们重写 equals()方法public boolean equals(Object object) {            if (object == null) {                return false;            }            if (object == this) {                return true;            }            //instanceof是判断该对象是否是该类的一个实例        if (!(object instanceof HashTest)) {                return false;            }            HashTest other = (HashTest) object;            if (other.getI() == this.getI()) {                return true;            }            return false;        }  经过这样的重写过后得到的结果就为true.