Java中==与equals()及hashcode()三者之间的关系详解

目录
  • 1.= =
  • 2.equals()
  • 3.重写equals()
  • 4.equals()比较流程
  • 5.hashcode()

1.= =

=为赋值运算符,==为比较运算符,仅比较对象的内存地址,无法比较真正意义上的相等!

JDK里的equals方法就是通过==来实现的比较对象的内存地址

以Integer为例

Integer a = 127;
Integer b = 127;
System.out.println(a == b);//true
Integer c = 128;
Integer d = 128;
System.out.println(c == d);//false

这里也是通过== 引出一个知识点,一个数值之差为啥导致结果不一样?在[-128,127]的区间内Integer a = 127;由于设计了缓存,而后的Integer b = 127;就是直接利用的缓存里的数值对象,所以通过==比较的结果为true,因为他们本质还是一个数值对象

而Integer c = 128;Integer d = 128;就没这样幸运了,超过了缓存区间会重新new出对象,以至于两者虽然数值相同但是地址不同

所以啊,==比较的是地址!也只是地址!

在前面==的基础上,再来看equals()

2.equals()

以一个String类型的变量为例,当我们来使用equals()比较两个对象时,结果肯定是false,因为equals()方法比较的是内存地址,这里的person1,person2均是两次new出来的,所以地址肯定是不相同的,而person1,person3指向同一空间地址一定是相同的

Person person1 = new Person("lyy");
Person person2 = new Person("lyy");
Person person3 = person1;
System.out.println(person1.equals(person2));//false
System.out.println(person1.equals(person3));//true

在不重写的情况下,我们Ctrl+B看一下equals()的源码:

public boolean equals(Object obj) {
    return (this == obj);
}

显而易见的是 (this = = obj)是该方法的核心,而 = = 又是两个对象比较的方式,= = 嘛比的是内存地址,懂的都懂噢

3.重写equals()

记不记得你在刷面经或者短视频的时候经常看到的一句话——比较两个对象的内容是否相等时我们要重写equals()方法!

这是为何?那我们不妨来重写一下equals()试试水

@Override
public boolean equals(Object obj){
    if (this==obj){
        return true;
    }
    if (obj==null||getClass()!= obj.getClass()){
        return false;
    }
 Person person=(Person)obj;
    return Objects.equals(name,person.name);
}

重写过equals后,原有的两者就已发生翻天覆地的变化,从原来的比较内存地址——>比较对象内容,这是一件很神奇的事情,因为实现了比较不同对象的相同或者不同内容!

重写之后:

具体是如何实现的呢?就像下面这样…

4.equals()比较流程

下面我们来探索一下重写的equals()是如何比较内容的:

通过debug来深入理解一下

下面来看一下debug过程中变量情况

总的来说,通过debug,重写equals()来比较不同变量的不同或者相同内容得到了进一步论证!

5.hashcode()

我们在IDEA中通过CTRL+O的快捷键重写hashcode()时它上面所属的类是谁?

java.lang.Object!

显而易见,该方法是Object类所定义的方法,作用是返回对象的哈希值返回值的类型为int(哈希值的作用是确定该对象在哈希表中的位置),曾经有这样一句流川千古的话:你必须在每个重写equals()的类中重写一遍hashcode()方法

如果不这样做将会违反Object.hashcode()的一般约定,这会阻止lei与所有基于散列的集合(比如hashmap,hashset…)一起正常工作。为啥?因为hashmap,hashset等基于散列的集合中,会使用对象的hashcode值来确定该对象应该如何存储到集合中,并且再次使用hashcode来定位对象在集合中的位置

那么

在一个类中重写了equals()但没重写hashcode()会出现啥情况呢?

我们来通过一个例子试试水~

尝试把对象都放入一个不能重复的set里,然后看集合的长度来判断两个对象是否相等!

public class equals_hashcode {
    public static void main(String[] args) {
        Person person1 = new Person("lyy");
        Person person2 = new Person("lyy");
        HashSet<Person> set = new HashSet<>();
        set.add(person1);
        set.add(person2);
        System.out.println(set.size());
    }
}
class Person {
    public String name;
    public Person(String name) {
        this.name = name;
    }
    @Override
    public boolean equals(Object obj) {
        if (this == obj) {
            return true;
        }
        if (obj == null || getClass() != obj.getClass()) {
            return false;
        }
        Person person = (Person) obj;
        return Objects.equals(name, person.name);
    }
    @Override
    public int hashCode() {
        return Objects.hash(name);
    }
}

情况一:

当只重写了equals()方法时运行的结果为2,由此得出两个对象不相等!

情况二:

当既重写了equals()和hashcode()后运行结果为1,所以两个对象相等!

由此得出,对象相等的本质是:

1.地址相同

2.哈希值相同(重写hashcode的体现)

这也不难联系到之前的约定了,在比较对象是否相等的场景下,我们必须重写equals()和hashcode()!

对象相等建立在==之上,equals(),hashcode()的双重写是对象相等的基本准则!

到此这篇关于Java中==与equals()及hashcode()三者之间的关系详解的文章就介绍到这了,更多相关Java == equals() hashcode()内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • 探索Java中的equals()和hashCode()方法_动力节点Java学院整理

    equals()和hashCode()区别?  equals():反映的是对象或变量具体的值,即两个对象里面包含的值--可能是对象的引用,也可能是值类型的值.  hashCode():计算出对象实例的哈希码,并返回哈希码,又称为散列函数.根类Object的hashCode()方法的计算依赖于对象实例的D(内存地址),故每个Object对象的hashCode都是唯一的:当然,当对象所对应的类重写了hashCode()方法时,结果就截然不同了. 之所以有hashCode方法,是因为在批量的对象比

  • Java基础之浅谈hashCode()和equals()

    写在前面 其实很早我就注意到阿里巴巴Java开发规范有一句话:只要重写 equals,就必须重写 hashCode. 我想很多人都会问为什么,所谓知其然知其所以然,对待知识不单止知道结论还得知道原因. hashCode方法 hashCode()方法的作用是获取哈希码,返回的是一个int整数 学过数据结构的都知道,哈希码的作用是确定对象在哈希表的索引下标.比如HashSet和HashMap就是使用了hashCode方法确定索引下标.如果两个对象返回的hashCode相同,就被称为"哈希冲突&quo

  • Java 中 hashCode() 与 equals() 的关系(面试)

    目录 一.基础:hashCode() 和 equals() 简介 equals() hashCode() 二. 漫谈:初识 hashCode() 与 equals() 之间的关系 三. 解密:深入理解 hashCode() 和 equals() 之间的关系 equals() 会有力不从心的时候 hashCode() 小力出奇迹 Java 设计 equals(),hashCode() 时约定的规则 四. 验证:结合 HashMap 的源码和官方文档,验证两者的关系 五. 结束 前言: Java 中

  • Java hashCode原理以及与equals()区别联系详解

    目录 1.什么是hashCode 2.equals()与hashCode()的联系 3.为什么重写equals()的同时要重写hashCode()方法 3.1.测试一 3.2.测试二 4.由hashCode()造成的内存泄露问题 5.基本数据类型和String类型的hashCode()方法和equals()方法 1.什么是hashCode hashCode就是对象的散列码,是根据对象的某些信息推导出的一个整数值,默认情况下表示是对象的存储地址.通过散列码,可以提高检索的效率,主要用于在散列存储结

  • java中重写equals()方法的同时要重写hashcode()方法(详解)

    object对象中的 public boolean equals(Object obj),对于任何非空引用值 x 和 y,当且仅当 x 和 y 引用同一个对象时,此方法才返回 true: 注意:当此方法被重写时,通常有必要重写 hashCode 方法,以维护 hashCode 方法的常规协定,该协定声明相等对象必须具有相等的哈希码.如下: (1) 当obj1.equals(obj2)为true时,obj1.hashCode() == obj2.hashCode()必须为true (2) 当obj

  • Java中==与equals()及hashcode()三者之间的关系详解

    目录 1.= = 2.equals() 3.重写equals() 4.equals()比较流程 5.hashcode() 1.= = =为赋值运算符,==为比较运算符,仅比较对象的内存地址,无法比较真正意义上的相等! JDK里的equals方法就是通过==来实现的比较对象的内存地址 以Integer为例 Integer a = 127; Integer b = 127; System.out.println(a == b);//true Integer c = 128; Integer d =

  • Java 中桥接模式——对象结构型模式的实例详解

    Java  中桥接模式--对象结构型模式的实例详解 一.意图 将抽象部分与它的实现部分分离,使他们都可以独立的变化. 二.适用性 以下一些情况使用Bridge模式 你不希望在抽象和它的实现部分之间有一个固定的绑定关系.例如这种情况可能因为,在程序运行时刻实现部分应可以被选择或者切换. 类的抽象以及它的实现都应该可以通过生成子类的方法加以扩充.这时Bridge模式使你可以对不同的抽象接口和实现部分进行组合,并分别对他们进行扩充. 对一个抽象的实现部分的修改应对客户不产生影响,即客户代码不必重新编译

  • java中的前++和后++的区别示例代码详解

    java中的前加加++和后加加++,有很多人搞的很晕,不太明白!今天我举几个例子说明下前++和后++的区别! 其实大家只要记住一句话就可以了,前++是先自加再使用而后++是先使用再自加! 前++和后++总结:其实大家只要记住一句话就可以了,前++是先自加再使用而后++是先使用再自加! 请大家看下面的例子就明白了! public class Test { public static void main(String[] args) { //测试,前加加和后加加 //前++和后++总结:其实大家只要

  • Java  中桥接模式——对象结构型模式的实例详解

    Java  中桥接模式--对象结构型模式的实例详解 一.意图 将抽象部分与它的实现部分分离,使他们都可以独立的变化. 二.适用性 以下一些情况使用Bridge模式 你不希望在抽象和它的实现部分之间有一个固定的绑定关系.例如这种情况可能因为,在程序运行时刻实现部分应可以被选择或者切换. 类的抽象以及它的实现都应该可以通过生成子类的方法加以扩充.这时Bridge模式使你可以对不同的抽象接口和实现部分进行组合,并分别对他们进行扩充. 对一个抽象的实现部分的修改应对客户不产生影响,即客户代码不必重新编译

  • java中synchronized(同步代码块和同步方法)详解及区别

     java中synchronized(同步代码块和同步方法)详解及区别 问题的由来: 看到这样一个面试题: //下列两个方法有什么区别 public synchronized void method1(){} public void method2(){ synchronized (obj){} } synchronized用于解决同步问题,当有多条线程同时访问共享数据时,如果进行同步,就会发生错误,Java提供的解决方案是:只要将操作共享数据的语句在某一时段让一个线程执行完,在执行过程中,其他

  • 关于java中可变长参数的定义及使用方法详解

    JAVA中可以为方法定义可变长参数( Varargs)来匹配不确定数量的多个参数,其定义用"..."表示.其实,这类似于为方法传了一个数组,且在使用方法上也和数组相同,如下: public void test(String... str){ for(String s : str){ } } 调用方法就像普通的调用方式相同,只不过可以匹配的参数为0到多个.如下: test(); test("lilei"); test("lilei","h

  • Java中SSM框架实现增删改查功能代码详解

    记录一下自己第一次整合smm框架的步骤. 参考博客和网站有:我没有三颗心脏 How2J学习网站 1.数据库使用的是mySql,首先创建数据库ssm1,并创建表student create database ssm1; use ssm1; CREATE TABLE student( id int(11) NOT NULL AUTO_INCREMENT, student_id int(11) NOT NULL UNIQUE, name varchar(255) NOT NULL, age int(1

  • JavaScript原型和原型链与构造函数和实例之间的关系详解

    目录 原型 原型链 原型 如图所示: 1.instanceof检测构造函数与实例的关系: function Person () {.........} person = new Person () res = person instanceof Person res  // true 2.实例继承原型上的定义的属性: function Person () {........} Person.prototype.type = 'object n' person = new Person () re

  • python中类与对象之间的关系详解

    在搜索平台上关于类以及对象都已经被霸屏了,主要的问题无非就是两个,一个是理解二者,另一个就是理解二者之间的使用关系,对于小编来说,两者统一跟大家讲清,相信也很难被大家消化,这不,给大家想出来比较好理解的方式,用最简单的话,快速交大家上手,可别不信,简单易懂内容如下. 二者关系: 女生口红是一种类,但是mac.完美日记是口红里的个体,被称作是对象.这就是二者之间的关系,有人理解成包含情况也可以. 定义类/对象: class 类名(父类): class Human(object): pass man

  • java集合——Java中的equals和hashCode方法详解

    Java中的equals方法和hashCode方法是Object中的,所以每个对象都是有这两个方法的,有时候我们需要实现特定需求,可能要重写这两个方法,今天就来介绍一些这两个方法的作用. equals()和hashCode()方法是用来在同一类中做比较用的,尤其是在容器里如set存放同一类对象时用来判断放入的对象是否重复. 这里我们首先要明白一个问题: equals()相等的两个对象,hashcode()一定相等,equals()不相等的两个对象,却并不能证明他们的hashcode()不相等.换

随机推荐