zoukankan      html  css  js  c++  java
  • 面试题:我们重写一个对象的时候为什么要同时重写hashcode()和equals()方法

    个人博客网:https://wushaopei.github.io/    (你想要这里多有)

    在创建的类不重写hashCode()和equals() 方法时,默认使用 java 提供的 java.lang.Object 下的 hashCode()和equals() 方法。
    > 注意:Object 的public boolean equals(Object obj)方法主要是对非空对象的引用地址的判断相同才返回true,而非对象本身的字符串内容或数值是否相同。

    简而言之,当且仅当 值A 和 值B 都是引用自同一个对象时,此方法才会返回true;

    所以,当我们重写一个对象,重写了equals()方法后,通常必须重写 hashCode()方法,以维护 hashCode 方法的常规协定,该协定声明了相等对象必须具有相等的哈希码。
    > 说白了,就是equals 返回true的两个值,在hashCode() 中结果也必然是true。

    *例子:*
    (1)当obj1.equals(obj2)为true时,obj1.hashCode() == obj2.hashCode()必须为true
    (2)当obj1.hashCode() == obj2.hashCode()为false时,obj1.equals(obj2)必须为false

    这里引出一个问题,为什么要重写equals()方法呢?难道原生的不能用吗?

    答案是必须重写,从前面的注意事项中可以知道,如果不重写equals 方法,那么比较的将是对象的引用是否指向同一块内存地址,而我们重写的*目的*是为了能够比较两个对象的value值是否相等。

    在高级类的八个包装类与引用类型的String类型(该类对euqals和hashcode方法进行了重写)中都 是使用重写后的 equals 方法来比较对象的,*默认*比较的是值,在*比较其它自定义对象*时都是比较的引用地址。

    后面会对equals 和 hashcode 的关系进行关联解释。

    *hashcode*是用于散列数据的快速存取,如利用*HashSet/HashMap/Hashtable*类来存储数据时,都是根据存储对象的*hashcode*值来进行判断是否*相同*的。

    那么,如果我们值重写equals ,而不重写 hashcode 会怎么样?

    如果我们对一个对象重写了equals 方法,意味着只要对象的成员变量值都相等那么equals 就返回true ,但在不重写 hashcode 方法的情况下,当我们重新 new 了一个新对象。
    此时,当原对象.equals(新对象)等于true时,两者的 hashcode 却是不一样的,由此会产生了理解的不一致,也违反了equals 与 hashcode的约定规则。

    不重写导致的结果案例:

    在不重写hashcode的情况下,如果 hashset存储两个引用不同但值相同的对象,此时hashcode返回false,认为后者与前者不重复,则会重新 newNode() 创建一个新节点将重复的值添加到集合中,意味着不可重复的单列集合中出现了两个值一样的对象,导致混淆。(假设没有重写)

    所以,需要重写 hashcode()方法。









  • 相关阅读:
    三元组数据结构
    线性表的顺序表示和实现 数据结构
    【欧拉计划1】Multiples of 3 and 5
    strcmp()与strcmpi()函数 C语言
    指向函数的指针 C语言
    const限定符声明 C语言
    Java环境搭建与配置
    栈的C语言实现
    【欧拉计划2】Even Fibonacci numbers
    单链表的表示和实现 数据结构
  • 原文地址:https://www.cnblogs.com/wushaopei/p/12283640.html
Copyright © 2011-2022 走看看