equals 与 hashCode 笔记二

第二篇文章主要讲解 hashCode 方法,分为以下部分:

  1. 关于 hashCode
  2. hashCode 源码
  3. 重写 hashCode 原因
  4. 如何重写 hashCode

关于 hashCode

Java中的集合(Collection)有两类,一类是 List,再有一类是 Set。前者集合内的元素是有序的,元素可以重复;后者元素无序,但元素不可重复。

两个元素是否重复应该依据什么来判断呢?

答案是 equals 方法。如果每增加一个元素就检查一次,在集合类的元素较少时,执行效率还算不错,但是那么当元素很多时,后添加到集合中的元素比较的次数就非常多了。也就是说,如果集合中现在已经有 1000 个元素,那么第 1001 个元素加入集合时,它就要调用 1000 次 equals 方法。这显然会大大降低效率。

一般情况下,计算机存储数据时,将数据连续地存储在内存单元中,于是,Java集合类在存储数据时,引进了hashCode方法,它采用了一种称为哈希算法,即将数据通过特定算法指定到某个内存单元。

当集合要添加新的元素时,先调用这个元素的hashCode方法,就一下子能定位到它应该放置的物理位置上。如果这个位置上没有元素,它就可以直接存储在这个位置上,不用再进行任何比较了;如果这个位置上已经有元素了,就调用它的 equals 方法与新元素进行比较,相同的话就不存了,不相同就散列其它的地址。所以这里存在一个冲突解决的问题。这样一来实际调用 equals 方法的次数就大大降低了,几乎只需要一两次。

所以,Java 对于 eqauls 方法和 hashCode 方法是这样规定的:

1.如果两个对象相同,那么它们的 hashCode 值一定要相同;
2.如果两个对象的 hashCode 相同,它们并不一定相同(这里说的对象相同指的是用 eqauls 方法比较)。

  1. equals() 相等的两个对象,hashCode() 一定相等;equals() 不相等的两个对象,却并不能证明他们的 hashCode() 不相等。

注: 规定 2 中指的是两个对象在哈希存储时发生了冲突。

hashCode 源码

public native int hashCode();

hashCode() 是一个本地方法,返回这个对象的哈希值,默认是返回该对象的内存地址。重写此方法可以提高哈希结构的集合的性能。

重写 hashCode 原因

对于 Java 集合类,我们经常使用 Set 集合来保存相关对象,而 Set 集合是不允许重复的。在向 HashSet 集合中添加元素时,其实只要重写 equals() 这一条也可以。但当 HashSet 中元素比较多时,或者是重写的 equals() 方法比较复杂时,我们只用 equals() 方法进行比较判断,效率也会非常低,所以引入了 hashCode() 这个方法,只是为了提高效率,且这是非常有必要的。

简单来说,hashCode存在的意义主要是提供查找的快捷性,比如说在Hashtable、HashMap中等。hashCode是用来在散列存储结构中确定对象存储的位置的;

如何重写 hashCode

重写 hashCode 所要遵循的原则如下:

  • 在程序执行期间,只要 equals 方法的比较操作用到的信息没有被修改,那么对这同一个对象调用多次,hashCode 方法必须始终如一地返回同一个整数
  • 如果两个对象通过 equals 方法比较得到的结果是相等的,那么对这两个对象进行hashCode得到的值应该相同
  • 两个不同的对象,hashCode 的结果可能是相同的,这就是哈希表中的冲突。为了保证哈希表的效率,哈希算法应尽可能的避免冲突

下面介绍如何来重写hashCode()方法。通常重写hashCode()方法按以下设计原则实现。

  1. 把某个非零素数,例如17,保存在int型变量result中。
  2. 对于对象中每一个关键域f(指equals方法中考虑的每一个域)参照以下原则处理。
  • boolean型,计算(f?0:1)。
  • byte、char和short型,计算(int)f。
  • long型,计算(int)(f^(f>>32))。
  • float型,计算Float.floatToIntBits(f)。
  • double型,计算Double.doubleToLongBits(f)得到一个long,再执行long型的处理。
  • 对象引用,递归调用它的hashCode()方法。
  • 数组域,对其中的每个元素调用它的hashCode()方法。
  1. 将上面计算得到的散列码保存到int型变量c,然后执行result = 37 * result + c,并返回。

根据上面的理解,我们进行相关测试。
Person类的 hashCode 重写如下:

@Override
 public int hashCode() {
    
    int result = 17;
    result = 37 * result + name.hashCode();
    return result;
}

Employee类的 hashCode 重写如下:

@Override
public int hashCode() {

   int result = 17;
   result = 37 * result + super.hashCode();
   result = 37 * result + id;
   return result;
}

测试类 HashCodeTest 源码如下:

public class HashCodeTest {

    public static void main(String[] args) {
        
        Employee e1 = new Employee("Mary", 18);
        Employee e2 = new Employee("Mary", 19);
        Person p1 = new Person("Mary");
        Person p2 = new Person("Mary");

        System.out.println("p1.equals(e1)'s rsult:" + p1.equals(e1));
        System.out.println("p1.equals(e2)'s rsult:" + p1.equals(e2));
        System.out.println("e1.equals(e2)'s rsult:" + e1.equals(e2));
        System.out.println("p1.equals(p2)'s rsult:" + p1.equals(p2));
        
        System.out.println("p1.hashCode is :" + p1.hashCode());
        System.out.println("p2.hashCode is :" + p2.hashCode());
        System.out.println("e1.hashCode is :" + e1.hashCode());
        System.out.println("e2.hashCode is :" + e2.hashCode());
    }

}

测试结果如下:

p1.equals(e1)'s rsult:false
p1.equals(e2)'s rsult:false
e1.equals(e2)'s rsult:false
p1.equals(p2)'s rsult:true
p1.hashCode is :2391408
p2.hashCode is :2391408
e1.hashCode is :88505387
e2.hashCode is :88505388

即 equals() 相等的两个对象,hashcode() 一定相等。创建一个类时,我们需要重写该类的 equals 和 hashCode 方法,若不对其进行重写,则会默认为 Object 类的 equals 和 hashCode 方法。

参考链接

  1. Java 中 equals() 与 hashCode() 方法详解
  2. hashCode 方法及 equals 方法的规范
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,724评论 6 479
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,104评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,142评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,086评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,076评论 5 370
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,914评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,220评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,871评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,318评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,834评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,951评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,574评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,162评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,162评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,383评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,349评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,652评论 2 343

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,579评论 18 399
  • 本文出自 Eddy Wiki ,转载请注明出处:http://eddy.wiki/interview-java.h...
    eddy_wiki阅读 1,152评论 0 16
  • 首先equals()和hashcode()这两个方法都是从object类中继承过来的。 很明显是对两个对象的地址值...
    itachi阅读 303评论 0 3
  • Java8张图 11、字符串不变性 12、equals()方法、hashCode()方法的区别 13、...
    Miley_MOJIE阅读 3,693评论 0 11
  • 2007-11-20 20:14 其实,我一直都记得你,记得你风中微笑的模样那样坦率那么纯净记忆里总有泛黄的角落悄...
    Helloe阅读 193评论 0 1