背景

我们知道在Java中比较两个对象是否相同,可以有多种方法,最常见的就是 == 和 equals 方法。但是由于==对比的是对象引用本身,因此重写equals才是最常用和可靠的比较两个对象是否相同的方法(当然重写equals同时意味着可能需要重写hashCode)

对于==和equals的异同,可以参考《阿里java规范》中对于Integer的一个很有意思的例子,因为Integer在实现时对-127~127的数字做了一个缓存,因此:

Integer i1 = 1, i2 = 1 ;

i1==i2; // true

i1.equals(i2); // true

Integer i1 = 1024, i2 = 1024 ;

i1==i2; // false

i1.equals(i2); // true

但是在实际搬砖的过程中,很多时候对于每个需要比较的对象都重写equals是很麻烦的(有时这种重写equals可能导致新的依赖情况),而且对于一些无法修改源码的类重写equals也是不可行的。

因此想到了通过反射来处理这样一些情况,就能比较简单的进行一些深度的比较,从而不需要重写每个比较对象的equals方法,也降低了编码时候的侵入性。

方案

因此基于以上想法,特别将这种利用反射来实现的深度比较封装了一个简单的功能类,能比较方便的实现对比的功能。所以这其实是一篇广告文,骗你去访问我的github的,没想到吧!哈哈哈!

重要的广告无耻的做完3次之后,更无耻的讲下怎么用好了!还是没想到吧,哈哈哈!

使用示例

具体的使用示例如下:

// 初始化并设置对比属性

CompareUtils compareUtils = CompareUtils.build();

compareUtils.ignoreAnnotation(); // 是否忽略自定的 @NotCompare 注解

compareUtils.ignoreCollection() // 是否忽略对Collection类型的属性的比较(e.g. List)

compareUtils.ignoreMap(); // 是否忽略对Map类型的属性的比较

// 直接使用isDifferent方法来比较得出是否是相同的对象

Boolean isDifferent = compareUtils.isDifferent(firstObject, secondObject);

当然对于初始化和设置属性,推荐使用这种方式:

CompareUtils utils = CompareUtils.build()

.includeAnnotation()

.includeCollection()

.includeMap();

具体属性设置说明

ignoreAnnotation/includeAnnotation : 鉴于灵活性的考虑,在对比中可能需要忽略一些不关注的属性值是否相同,因此提供了一个 @NotCompare 的注解,缺省情况下会跳过使用了注解的属性,但是可以通过该设置来选择是否开启跳过注解这一功能;

ignoreCollection/includeCollection : 缺省情况下会对比对象中Collection类型的属性,但是可以通过该方法来开启或关闭这一行为

ignoreMap/includeMap : 缺省情况下会对比对象中Map类型的属性,但是可以通过该方法来开启或关闭这一行为

注意点

对比的对象 不能有递归嵌套

通过一些会 增加属性数量 的方法生成的对象(比如Mokito.mock的对象),会造成 比对结果不准确,因为工具类在缺省情况下会将所有的属性都进行比较,而增加属性数量方法生成的对象,可能包含了一些额外的不相同的属性,造成本应该相同的对象最后存在不同的情况

java比较两个对象重写,不重写equals进行两个对象间的深度比较相关推荐

  1. HashMap存自定义对象为什么要重写 hashcode 和 equals 方法?

    HashMap的k放过自定义对象么? 当我们把自定义对象存入HashMap中时,如果不重写hashcode和equals这两个方法,会得不到预期的结果. class Key{private Integ ...

  2. java全栈系列之JavaSE-面向对象(方法重写)037

    子类在调用父类的私有方法中不能直接调用,但是可以通过get方法进行调用,修改属性的值可以通过set方法进行修改.而子类想要修改父类中的方法可以使用方法重写进行操作. 方法重写与之前的方法重载不同 回顾 ...

  3. java继承对象转换_java中类与对象的继承重写,存储以及自动转换和强制转换。...

    对象的继承 继承关键字:extends 继承的格式: public class 类名 extends 父类名{ } 注:一个类只能继承一个父类.子类继承父类的全部内容. 访问修饰符同类中 同包中 不同 ...

  4. Java中关于==和equal的区别 以及equals()方法重写

    例子i: string1="aaa"; string2="aaa"; String string3=new String("aaa"); S ...

  5. java重写的目的是什么_什么是重写?重写的作用?

    一.什么是重写? 1.重写(Override)概念 重写是子类对父类的允许访问的方法的实现过程进行重新编写, 返回值和形参都不能改变.即外壳不变,核心重写! 重写的好处在于子类可以根据需要,定义特定于 ...

  6. Java面向对象 继承,super,方法重写

    /* 继承:表示父类跟子类之间的关系当两个类或者多个类具备相同的属性和方法的时候,可以提取出来,变成父类,子类可以继承子类跟父类是is-a的关系使用:1.使用继承的时候需要使用extend关键字2.使 ...

  7. Java基础学习第九节——继承、重写

    面向对象--继承 1. 继承 1.1 继承概念的引入 继承是面向对象最显著的一个特性.继承是从已有的类中派生出新的类,新的类能吸收已有类的数据属性和行为,并能扩展新的能力. 代码实现: 三个类 都有重 ...

  8. 面向对象 详解笔记 方法 类 对象 封装 继承 重写 多态 接口

    文章目录 面向过程&面向对象 面向过程 面向对象 什么是面向对象 回顾方法及加深 方法的定义 修饰符 返回类型 break:跳出switch,结束循环和return的区别 方法名:注意规范就o ...

  9. java中equals的重写_Java重写equals方法(重点讲解)

    为什么equals()方法要重写? 判断两个对象在逻辑上是否相等,如根据类的成员变量来判断两个类的实例是否相等,而继承Object中的equals方法只能判断两个引用变量是否是同一个对象.这样我们往往 ...

  10. Java基础系列:重写hashCode和equals

    1 场景 Map的key设置为对象时,必须重写对象的hashCode和equals方法. 原因 通过对象取值时,及时相同的对象(初始化值相同),get时,会出现null值. 方案 重写对象的hashC ...

最新文章

  1. IntelliJ IDEA 最常用配置,应用、永久激活
  2. vscode 将本地项目上传到github、从github克隆项目以及删除github上的某个文件夹
  3. 《Python参考手册(第4版•修订版)》——1.4 文件输入和输出
  4. 算法学习笔记:对指定金额计算最少钞票数
  5. com.alibaba.fastjson.JSONException: can‘t create non-static inner class inst
  6. v3是c语言吗 yolo_你真的明白yolo v3吗?
  7. word简历排版技巧
  8. c语言编程中句柄无效怎么解决,Win7玩英雄联盟出现句柄无效怎么解决?
  9. 重庆2021年高考二诊成绩查询,2021年重庆二诊,2021年4月重庆二诊考试,重庆二诊康德卷...
  10. <数据结构>链表实战之单链表与双链表的增删改查
  11. UVA1449 Dominating Patterns
  12. 苹果11怎么录屏_苹果11突然黑屏是怎么回事?
  13. Rebuild Project
  14. AI中插入带圆圈的1-20数字
  15. 列表(list)使用方法详解
  16. matlab怎么看输出电压纹波,Boost变换器的能量传输模式和输出纹波电压分析.pdf
  17. MobaXterm 最下面显示服务器信息
  18. 五大爆款单视频开头,学会做短视频很轻松
  19. 32位与64位操作系统的区别
  20. 开源中国源码学习(一)——简介

热门文章

  1. 高仿真的类-DefaultListableBeanFactory
  2. AOP 代理(AOP Proxy)
  3. 支付宝支付 - 构建支付表单填并提交
  4. 静态工厂配置bean
  5. 选择结构_单if语句
  6. SpringBoot_配置-外部配置加载顺序
  7. 内核aio_今天来说说令人让人傻傻分不清的BIO,NIO,AIO
  8. pytorch教程龙曲良26-30
  9. 201119阶段二sqlite3
  10. 201105阶段二qt创建简单工程