热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

几分钟理解Jdk==,hashCode()与equals()

小概,hashCode()与equals(),这三个操作在Java程序当中满地都是,特别是容器里,如Map中的哈希映射与搜索元素就是根据h

小概

==,hashCode() 与 equals() ,这三个操作在 Java 程序当中满地都是,特别是容器里,如 Map 中的哈希映射与搜索元素就是根据
hashCode() 和 equals() 判断的,所以如何正确的理解和使用显得非常重要,并且在封装类的时候,十分建议通通重写

我们先要知道这个概念,每个线程都有属于自己的 虚拟机栈,虚拟机栈中的元素我们称之为 栈帧,每运行一个方法时,虚拟机会为这个方法创建一个栈帧,并入栈,方法结束后便出栈

那么我们操作的变量有两种

  • 局部变量:存在当前栈帧的局部变量表里,如果是基本数据类型便是值的大小,如果是对象便是一个指向堆内存中对象地址的引用

  • 全部变量:存在方法区中,存储规则一样

所以我们比较的,是 基本数据类型的大小引用地址

a == b

比较规则

  1. 当a,b 为对象时,表示两者 引用地址 是否相同,即堆内存中地址是否相同

  2. 当a,b 为基本数据类型时,表示两者 数据大小 是否相等

a.equals(b)

假设 a 的对象类型为 A

比较规则

  1. a,b 只能为对象,表示两者 对象属性 是否相等,如 String 就是比较内部维护的 char[] 数组每一个字符是否相等

  2. 为了简化比较复杂度,往往会先判断 a == b,如果它们都指向同一个地址,那么两者内容肯定一样,则不需要逐一比较对象内容中的值

我们可以认为,== 和 equals 的推导关系是 充分不必要

  • == 成立则 equals 成立

  • equals 成立,== 不一定成立

如何比较属性

比如区分两个学生是否是同一个学生,我们其实只需要查看他们的学号是否相同就行了

也就是我们应该比较能唯一标识这个对象的一个甚至多个对象属性,情况完全视不同对象而定

如果 A 未重写 equals()

如果 A 未重写时,调用的是父类的 equals(),还未重写,会检查到 Object,而 Object 是如下比较的

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

利用 == 号比较对象内容是否相等未必有点草率,因此我们对这个方法应当十分谨慎

如何重写

重写时需注意应当满足如下规则,摘自 《Effective Java》

  1. 自反性:x.equals(x)必须返回true

  2. 对称性:x.equals(y)与y.equals(x)的返回值必须相等

  3. 传递性:x.equals(y)为true,y.equals(z)也为true,那么x.equals(z)必须为true

  4. 一致性:如果对象x和y在equals()中使用的信息都没有改变,那么x.equals(y)值始终不变

  5. 非null:x不是null,y为null,则x.equals(y)必须为false

在这里建议按以下规则重写

  1. 建议判断 a == b,省去没必要的比较

  2. 建议先判断 b instanceOf A,再做能能标识唯一对象的一个或多个属性的比较

  3. 建议考虑 super.equals(b)

a.hashCode()

该方法返回的是对象的哈希值,主要应用于哈希容器映射时,将所有类型对象映射成整数

如何重写

在这里建议按以下规则重写

  1. 建议对能标识唯一对象的一个或多个属性进行 hashCode() 重写

  2. 建议利用 jdk 基础封装类的 hashCode()

  3. 建议考虑 super.equals(b)




推荐阅读
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • 本文介绍了如何在给定的有序字符序列中插入新字符,并保持序列的有序性。通过示例代码演示了插入过程,以及插入后的字符序列。 ... [详细]
  • 猜字母游戏
    猜字母游戏猜字母游戏——设计数据结构猜字母游戏——设计程序结构猜字母游戏——实现字母生成方法猜字母游戏——实现字母检测方法猜字母游戏——实现主方法1猜字母游戏——设计数据结构1.1 ... [详细]
  • [大整数乘法] java代码实现
    本文介绍了使用java代码实现大整数乘法的过程,同时也涉及到大整数加法和大整数减法的计算方法。通过分治算法来提高计算效率,并对算法的时间复杂度进行了研究。详细代码实现请参考文章链接。 ... [详细]
  • Java在运行已编译完成的类时,是通过java虚拟机来装载和执行的,java虚拟机通过操作系统命令JAVA_HOMEbinjava–option来启 ... [详细]
  • 集合的遍历方式及其局限性
    本文介绍了Java中集合的遍历方式,重点介绍了for-each语句的用法和优势。同时指出了for-each语句无法引用数组或集合的索引的局限性。通过示例代码展示了for-each语句的使用方法,并提供了改写为for语句版本的方法。 ... [详细]
  • 本文详细解析了JavaScript中相称性推断的知识点,包括严厉相称和宽松相称的区别,以及范例转换的规则。针对不同类型的范例值,如差别范例值、统一类的原始范例值和统一类的复合范例值,都给出了具体的比较方法。对于宽松相称的情况,也解释了原始范例值和对象之间的比较规则。通过本文的学习,读者可以更好地理解JavaScript中相称性推断的概念和应用。 ... [详细]
  • C语言注释工具及快捷键,删除C语言注释工具的实现思路
    本文介绍了C语言中注释的两种方式以及注释的作用,提供了删除C语言注释的工具实现思路,并分享了C语言中注释的快捷键操作方法。 ... [详细]
  • 1,关于死锁的理解死锁,我们可以简单的理解为是两个线程同时使用同一资源,两个线程又得不到相应的资源而造成永无相互等待的情况。 2,模拟死锁背景介绍:我们创建一个朋友 ... [详细]
  • Android JSON基础,音视频开发进阶指南目录
    Array里面的对象数据是有序的,json字符串最外层是方括号的,方括号:[]解析jsonArray代码try{json字符串最外层是 ... [详细]
  • 标题: ... [详细]
  • 前景:当UI一个查询条件为多项选择,或录入多个条件的时候,比如查询所有名称里面包含以下动态条件,需要模糊查询里面每一项时比如是这样一个数组条件:newstring[]{兴业银行, ... [详细]
author-avatar
书友78403557
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有