美文网首页Java 杂谈
你有没有想过: Java 虚拟机是如何判断两个对象是否相同的?判

你有没有想过: Java 虚拟机是如何判断两个对象是否相同的?判

作者: java高级编程中心 | 来源:发表于2019-02-22 21:55 被阅读1次

Java 虚拟机会先判断两个对象的hashCode是否相同,如果hashCode不同,则说明肯定是两个不同的对象了;如果hashCode相同再通过equals()方法进行进一步比较,如果equals方法返回true,则说明两个对象是相同的,如果equals方法返回false说明两个对象不同。

如何验证

怎么来验证这个问题呢?我们知道HashSet是不允许存储相同的键值的。所以我们可以用HashSet存储两个相同的键值来模拟,看Java虚拟机是如何做判断和识别的,从而验证我们的猜想。

// 先自定义一个类并复写 hashCode 和 equals 方法

public class CustomClass {

    @Override

    public int hashCode() {

        System.out.println("判断 hashCode");

        return 1; // 返回1,说明所有新建的对象的哈希值都为1,也就是相同

    }

    @Override

    public boolean equals(Object o) {

        System.out.println("判断 equals");

        return true; // 返回true

    }

}

接下来我们用HashSet来存储两个自定义的CustomClass的对象,代码如下:

public class HashSetTest {

    public static void main(String[] args) {

        HashSet<CustomClass> hs = new HashSet<>();

        CustomClass cs1 = new CustomClass();

        CustomClass cs2 = new CustomClass();

        hs.add(cs1);

        hs.add(cs2);

        System.out.println("----hs添加完毕");

        System.out.println("hs:"+hs); // 打印一下hashSet集合看里面存放了什么

    }

}

打印结果如下:

判断 hashCode

判断 hashCode

判断 equals

----hs添加完毕

判断 hashCode // 此处的判断是打印输出语句执行时调用的,与分析本问题无关

hs:[com.alankeene.javalib.collections.CustomClass@1]

结果分析:

执行hs.add(cs1)语句的时候,Java 虚拟机会先判断 cs1 所指向对象的hashCode,因为是第一次往HashSet集合里面存放元素,该元素 hashCode 在集合中肯定是还没存在的,这是个新的元素,所以直接存放进集合中,不用调用 cs1 所指向对象的 equals 方法。

当执行hs.add(cs2)语句时,这是第二次往集合里存放元素,有新的元素 cs2 要添加进来,那先要调用 cs2 所指向对象的 hashCode 方法看看它的哈希值是不是与集合中已有元素的哈希值重复了,发现重复了,哈希值都是1,那有可能是同一个对象,那就要调用 cs2 所指向对象的 equals 方法做进一步判断,发现 equals 方法返回 true,则判断为是重复的元素,就不往集合里添加了。

所以最终打印 HashSet 集合的时候可以看到,集合中只存放了一个元素。

我们再反证一下,把 equals 方法改为返回 false,模拟两个hashCode相同,但是是两个不同的对象的情景。

public class CustomClass {

    @Override

    public int hashCode() {

        System.out.println("判断 hashCode");

        return 1; // 返回1,说明所有新建的对象的哈希值都为1,也就是相同

    }

    @Override

    public boolean equals(Object o) {

        System.out.println("判断 equals");

        return false; // 返回false

    }

}

打印结果会如下:

判断 hashCode

判断 hashCode

判断 equals

----hs添加完毕

判断 hashCode // 此处的判断是打印输出语句执行时调用的,与分析本问题无关

判断 hashCode // 此处的判断是打印输出语句执行时调用的,与分析本问题无关

hs:[com.alankeene.javalib.collections.CustomClass@1, com.alankeene.javalib.collections.CustomClass@1]

会发现,HashSet集合中存放了两个元素了,说明虽然 cs1 和 cs2 的哈希值相同,但是虚拟机判断为不同的元素并存入集合中了。

由此,验证了我们的猜想。

小编整理了一些java进阶学习资料和面试题,需要资料的请加JAVA高阶学习Q群:664389243 这是小编创建的java高阶学习交流群,加群一起交流学习深造。群里也有小编整理的2019年最新最全的java高阶学习资料!

相关文章

  • 你有没有想过: Java 虚拟机是如何判断两个对象是否相同的?判

    Java 虚拟机会先判断两个对象的hashCode是否相同,如果hashCode不同,则说明肯定是两个不同的对象了...

  • 哈希表

    哈希表确定元素是否相同 1. 判断的是两个元素的哈希值是否相同 如果相同,再判断两个对象的内容是否相同。 2. 判...

  • String中equals方法

    Java中使用“==”来判断两个对象是否相等是判断两个对象的引用是否相同Java中使用“equals”来判断两个对...

  • Java中的== 和equals 以及 kotlin中的==和=

    Java == 比较的是变量(栈)内存中存放的对象的(堆)内存地址,用来判断两个对象的地址是否相同,即是否是指相同...

  • 深入学习JVM 【3】判断一个对象是否可回收

    我们知道,虚拟机判断一个对象是否“已死”,是判断对象是否还有引用指向它。而虚拟机又是如何判断是否有引用指向对象呢?...

  • Java重写equals,hashCode和散列关系

    Java判定对象相同 equals方法是用来判断两个对象是否"相同":如果没有进行重写 那么默认默认情况下equa...

  • 112.好用常用工具类收集

    判断是否是空对象,可以判判断 对象和数组 持续更新......

  • 面试题

    1.如何判断两对象相等? isEqual:先判断指针两个对象的指针是否相同,若相同则为同一对象,若不同,比较cla...

  • java的四种引用

    Java对引用的定义 无论是通用引用计数算法判断对象的引用数据,还是通过可达性分析算法判断对象的引用链是否可达,判...

  • Java对象——死或生

    本文是《深入理解Java虚拟机》3.2节的读书笔记,理解有误的地方欢迎指正 如何判断对象是否可以回收,传统的做法是...

网友评论

    本文标题:你有没有想过: Java 虚拟机是如何判断两个对象是否相同的?判

    本文链接:https://www.haomeiwen.com/subject/spmyyqtx.html