Java:判断是否相等小结

对 Java 中的判断是否相等,即判断两数/两对象是否相等,做一个微不足道的小小小小记

== 判断

对于基本类型和引用类型 == 的效果是不同的,如下:

  • 基本类型:比较的值是否相同
  • 引用类型:比较的是是否引用同一对象,即引用对象的地址值是否相等

示例:

int int_a = 10;
int int_b = 10;
System.out.println(int_a==int_b); // true String stra = "abc";
String strb = "abc";
String strc = new String("abc"); // 通过new String()方法重新开辟了内存空间
String strd = new String("abc");
System.out.println(stra==strb); // true
System.out.println(stra==strc); // false
System.out.println(strc==strd); // false

equals 方法

equals() 方法:首先,原始的 equals 方法,只是用于比较两个对象的内存地址是否为同一个,如下:

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

而如果没有对 equals 方法进行重写,则比较的是引用类型的变量所指向的对象的地址(很多类重写了 equals 方法,比如 String、Integer 等把它变成了值比较,所以一般情况下 equals 比较的是值是否相等)。

如下代码所示:

// Person没有对equals方法进行重新,而String对equals方法进行了重写
private class Person{
public String name; public Person(String name) {
this.name = name;
}
} @Test
public void test02(){
String strc = new String("abc");
String strd = new String("abc");
System.out.println(strc.equals(strd)); // true Person p1 = new Person("张三");
Person p2 = new Person("张三");
System.out.println(p1.equals(p2)); // false
}

查看 Person 类中的 equals 方法:本质上 equals 就是 ==,即比较两个对象地址

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

查看 String 类中的 equals 方法:

public boolean equals(Object anObject) {
if (this == anObject) {
return true; // 地址相同直接返回true
}
if (anObject instanceof String) { // 判断是否为String对象
String anotherString = (String)anObject;
int n = value.length;
if (n == anotherString.value.length) { // 长度相同才进行比较
char v1[] = value;
char v2[] = anotherString.value;
int i = 0;
while (n-- != 0) { // 按字符比较
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}

同时需要注意的是:equals 不能用于基础类型的比较,因为 equals 是一个方法,而基础类似并不是一个对象,没有对应的方法

hashCode() 方法

hashCode() 方法:

提到 hashCode,自然联想到哈希表,通过 hashCode 方法,将 key 映射到哈希表中的一个位置,从而达到最好情况下以 O(1) 的时间复杂度来查询。

// Object类的源码中,hashcode() 是一个 native 方法,哈希值的计算利用的是内存的地址
public native int hashCode();

问:为什么重写 equals() 就一定要重写 hashCode() 方法?

equals() 方法:

首先,原始的 equals 方法,只是用于比较两个对象的内存地址是否为同一个,如下:

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

而重写 equals() 方法,是为了实现当两个对象指向的内存地址相同或者两个对象各自字段值相同,那么标识其为同一个对象

为什么要重写?

这个问题应该是有个前提,就是你需要用到 HashMap、HashSet 等 Java 集合,用不到哈希表的话,其实仅仅重写 equals() 方法也可以。而工作中的场景是常常用到 Java 集合,所以 Java 官方建议重写 equals() 就一定要重写 hashCode() 方法。

  1. 提高效率:对于对象集合的判重,如果一个集合含有 10000 个对象实例,仅仅使用 equals() 方法的话,那么对于一个对象判重就需要比较 10000 次,随着集合规模的增大,时间开销是很大的。但是同时使用哈希表的话,就能快速定位到对象的大概存储位置,并且在定位到大概存储位置后,后续比较过程中,如果两个对象的 hashCode 不相同,也不再需要调用 equals() 方法,从而大大减少了 equals() 比较次数
  2. 在实际应用过程中,如果仅仅重写了 equals(),而没有重写 hashCode() 方法,会出现什么情况?字段属性值完全相同的两个对象因为 hashCode 不同,所以在 HashMap 中的 table 数组的下标不同,从而这两个对象就会同时存在于集合中,所以重写 equals() 就一定要重写 hashCode() 方法。

hashCode()equals() 的相关规定:

  1. 如果两个对象相等,则 hashCode 一定也是相同的;
  2. 两个对象相等,对两个对象分别调用 equals 方法都返回 true;
  3. 两个对象有相同的 hashCode 值,它们也不一定是相等的;
  4. equals 方法被覆盖过,则 hashCode 方法也必须被覆盖;
  5. hashCode() 的默认行为是对堆上的对象产生独特值。如果没有重写 hashCode(),则该 class 的两个对象无论如何都不会相等(即使这两个对象指向相同的数据)。

对上述第3点进行说明:两个对象的 hashCode() 相同,则 equals() 不一定为 true,如下

@Test
public void test03(){
String str1 = "通话";
String str2 = "重地"; System.out.println("str1 hashCode:"+str1.hashCode()); // 1179395
System.out.println("str2 hashCode:"+str2.hashCode()); // 1179395
}

在散列表中,hashCode()相等,只是说明两个键值对的哈希值相等,但是哈希值相等,并不一定可以得出键值对相等。

参考

https://mp.weixin.qq.com/s/4E3xRXOVUQzccmP0yahlqA

https://blog.csdn.net/qq_41701956/article/details/103223461

https://blog.csdn.net/sixingmiyi39473/article/details/78306296

Java:判断是否相等小记的更多相关文章

  1. Java:ConcurrentHashMap类小记-3(JDK8)

    Java:ConcurrentHashMap类小记-3(JDK8) 结构说明 // 所有数据都存在table中, 只有当第一次插入时才会被加载,扩容时总是以2的倍数进行 transient volat ...

  2. Java:ConcurrentHashMap类小记-2(JDK7)

    Java:ConcurrentHashMap类小记-2(JDK7) 对 Java 中的 ConcurrentHashMap类,做一个微不足道的小小小小记,分三篇博客: Java:ConcurrentH ...

  3. Java:HashMap类小记

    Java:HashMap类小记 对 Java 中的 HashMap类,做一个微不足道的小小小小记 概述 HashMap:存储数据采用的哈希表结构,元素的存取顺序不能保证一致.由于要保证键的唯一.不重复 ...

  4. Java:HashTable类小记

    Java:HashTable类小记 对 Java 中的 HashTable类,做一个微不足道的小小小小记 概述 public class Hashtable<K,V> extends Di ...

  5. Java:LinkedList类小记

    Java:LinkedList类小记 对 Java 中的 LinkedList类,做一个微不足道的小小小小记 概述 java.util.LinkedList 集合数据存储的结构是循环双向链表结构.方便 ...

  6. Java:ArrayList类小记

    Java:ArrayList类小记 对 Java 中的 ArrayList类,做一个微不足道的小小小小记 概述 java.util.ArrayList 是大小可变的数组的实现,存储在内的数据称为元素. ...

  7. Java:动态代理小记

    Java:动态代理小记 对 Java 中的 动态代理,做一个微不足道的小小小小记 概述 动态代理:当想要给实现了某个接口的类中的方法,加一些额外的处理.比如说加日志,加事务等.可以给这个类创建一个代理 ...

  8. 使用Java判断字符串中的中文字符数量

    Java判断一个字符串str中中文的个数,经过总结,有以下几种方法(全部经过验证),可根据其原理判断在何种情况下使用哪个方法: 1. char[] c = str.toCharArray(); for ...

  9. Java判断回文数算法简单实现

    好久没写java的代码了, 今天闲来无事写段java的代码,算是为新的一年磨磨刀,开个头,算法是Java判断回文数算法简单实现,基本思想是利用字符串对应位置比较,如果所有可能位置都满足要求,则输入的是 ...

  10. Java 判断操作系统类型(适用于各种操作系统)

    Java 判断操作系统类型(适用于各种操作系统) 最近一段时间写一个授权的程序,需要获取很多信息来保证程序不能随意复制使用,必须经过授权才可以. 为了限制用户使用的操作系统,必须有统一的方法来获取才可 ...

随机推荐

  1. Qt 程序发布以及打包成exe安装包

    一.简述 Qt 项目开发完成之后,需要打包发布程序,而因为用户电脑上没有 Qt 配置环境,所以需要将 release 生成的 exe 文件和所依赖的 dll 文件复制到一个文件夹中,然后再用 Inno ...

  2. Tars | 第7篇 TarsJava Subset最终代码的测试方案设计

    目录 前言 1. SubsetConf配置项的结构 1.1 SubsetConf 1.2 RatioConfig 1.3 KeyConfig 1.4 KeyRoute 1.5 SubsetConf的结 ...

  3. HearthbuddyHelper已经开源

    https://github.com/V-arc/HearthbuddyHelper 只是为了稳定,没耗时多久写的一个中控,只是在逐渐堆加,并未进行重构. 通过写这个感受到自己对异步和wpf的理解还有 ...

  4. EXCEL中给包含某个字段的单元格所在行标注颜色

    条件格式->新建规则->使用公式确定要设置格式的单元格 公式: =COUNTIF($A1,"*字符串*") 注意:通配符的使用.

  5. 使用@EnableConfigurationProperties注册配置Bean时的命名规则

    Spring和Spring Boot开发中,常使用@ConfigurationProperties注解某个类,使其实例接受一组具有相同前缀的配置项. 可以使用@Component或Java Confi ...

  6. 洛谷P1308——单词统计

    https://www.luogu.org/problem/show?pid=1308 题目描述 一般的文本编辑器都有查找单词的功能,该功能可以快速定位特定单词在文章中的位置,有的还能统计出特定单词在 ...

  7. javascript 字符串反转 strip_tags 字符串常用的自定义函数,加载css etc.

    字符串反转 String.prototype.reverse = function() { var a = this.split(''); for (var i = 0, j = a.length-1 ...

  8. 【大咖直播】Elastic 企业搜索实战工作坊(第一期)

    借助 App Search 提供的内置功能,您可轻松打造卓越的搜索体验.直观的相关度调整以及开箱即用的搜索分析,不仅可以优化所提供的内容,其提供的 API 还可帮助您将位于各处的所有内容源关联在一起. ...

  9. 鸿蒙内核源码分析(工作模式篇) | CPU是韦小宝,七个老婆 | 百篇博客分析OpenHarmony源码 | v36.04

    百篇博客系列篇.本篇为: v36.xx 鸿蒙内核源码分析(工作模式篇) | CPU是韦小宝,七个老婆 | 51.c.h .o 硬件架构相关篇为: v22.xx 鸿蒙内核源码分析(汇编基础篇) | CP ...

  10. P4755-Beautiful Pair【笛卡尔树,线段树】

    正题 题目链接:https://www.luogu.com.cn/problem/P4755 题目大意 \(n\)个数字的一个序列,求有多少个点对\(i,j\)满足\(a_i\times a_j\le ...