• Javers 比较两个类的差异


    Javers

    在开发过程中遇到需求,比较数据库中的原数据与新修改要写入库中的数据。这个实体类是比较复杂的。例如有基本类型,BigDecimal类型,自定义类型,Date类型,List集合,Set集合,Map集合等。在比较复杂的类的情况下,使用Objects的equals是不适用的。所以为了解决这个问题,可以使用Javers库中的compare方法来解决该问题

    引入

     <!-- https://mvnrepository.com/artifact/org.javers/javers-core -->
            <dependency>
                <groupId>org.javers</groupId>
                <artifactId>javers-core</artifactId>
                <version>6.6.2</version>
            </dependency>
    

    方法:

    Javers javers = JaversBuilder.javers().build();
    Diff diff = javers.compare(oldObject,newObject)
    boolean flag = diff.hasChange();
    //diff会在控制台打印出两个类之间的异同,相同不打印数据,不同会展示出修改前后的数据
    //diff.hasChange(); 如果两个类完全相同,返回值为flase,两个类不同,返回值为true
    

    示例

    public static void main(String[] args) {
            Javers build = JaversBuilder.javers().build();
    	    //A系统的类
        	//数据库中的数据
            PromotionDetail promotionDetail = new PromotionDetail();
            promotionDetail.setCanEditFlag(true);
            promotionDetail.setAllGoodsCountType(1);
            promotionDetail.setName("name1");
            PromotionGoods promotionGoods = new PromotionGoods();
            promotionGoods.setCount(1);
            promotionGoods.setDeptName("deptname1");
            PromotionGoods promotionGoods1 = new PromotionGoods();
            promotionGoods1.setCount(2);
            promotionGoods1.setDeptName("deptname2");
            List<PromotionGoods> list = new ArrayList<>();
            list.add(promotionGoods);
            list.add(promotionGoods1);
            promotionDetail.setGoods(list);
    
    		//新数据
            PromotionDetail promotionDetail1 = new PromotionDetail();
            promotionDetail1.setCanEditFlag(true);
            promotionDetail1.setAllGoodsCountType(1);
            promotionDetail1.setName("name2");
            PromotionGoods promotionGoods2 = new PromotionGoods();
            promotionGoods2.setCount(5);
            promotionGoods2.setDeptName("deptname3");
            PromotionGoods promotionGoods3 = new PromotionGoods();
            promotionGoods3.setCount(4);
            promotionGoods3.setDeptName("deptname3");
            List<PromotionGoods> list0 = new ArrayList<>();
            list0.add(promotionGoods2);
            list0.add(promotionGoods3);
            promotionDetail1.setGoods(list0);
    
    
    
            Diff compare = build.compare(promotionDetail, promotionDetail1);
            boolean b = compare.hasChanges();
            System.out.println(compare);
            System.out.println(b);
        }
    
    //控制台输出
    Diff:
    * changes on com.ef.admin.data.controller.promotion.np.param.PromotionDetail/ :
      - 'goods/0.count' changed: '1' -> '5'
      - 'goods/0.deptName' changed: 'deptname1' -> 'deptname3'
      - 'goods/1.count' changed: '2' -> '4'
      - 'goods/1.deptName' changed: 'deptname2' -> 'deptname3'
      - 'name' changed: 'name1' -> 'name2'
          
    true   ====》为true说明有改变
    

    遇到的问题:

    • BigDecimal

      在比较BigDecimal类型的不同数据时,会出现问题。由于javer.compare()底层是使用的objects的equals方法来判断两个数据是否相等,而BigDecimal类型比较时使用equals方法比较存在问题,应该使用BigDecimal类中的compareTo方法比较,详见https://blog.csdn.net/molihuaya/article/details/79139418

      BigDecimalComparatorWithFixedEquals类实现了CustomValueComparator<>接口并重写equals方法

      public class BigDecimalComparatorWithFixedEquals implements CustomValueComparator<BigDecimal> {
          @Override
          public boolean equals(BigDecimal a, BigDecimal b) {
              return a.compareTo(b) == 0;
          }
      

      所以创建javers对象时:

      Javers javers = JaversBuilder.javers().registerValue(BigDecimal.class,new BigDecimalComparatorWithFixedEquals()).build();
      
    • Date

      在比较Date类型的不同数据时,会出现问题。假设在数据中查出的日期类型为Date,新修改的数据为TimeStamp类型。此时使用javers.compare()比较时即使时间点相同,比较出来仍有问题。

      数据库中的数据  2022-02-25 23:59:59.0
      新修改的数据    Fri Feb 25 23:59:59 CST 2022
      

      可以通过实现CustomValueComparator<>接口来重写equals方法

      public class EfDateComparator implements CustomValueComparator<Date> {
          @Override
          public boolean equals(Date a, Date b) {
              return a.getTime() == b.getTime();
          }
      

      所以创建javers对象时:

      Javers javers = JaversBuilder.javers().registerValue(Date.class,new EfDateComparator ()).build();
      
    Javers javers = JaversBuilder.javers()
        .registerValue(BigDecimal.class,new BigDecimalComparatorWithFixedEquals())
        .registerValue(BigDecimal.class,new BigDecimalComparatorWithFixedEquals())
        .build();
    

    相关注解

    在比较两个类时,可能需要只比较部分字段或不比较部分字段,可以考虑使用以下两个注解

    //在一个类中,在字段上添加该注解,该类中只比较有注解的字段
    @DiffInclude
    
    //在一个类中,在字段上添加该注解,该类中忽略比较有注解的字段
    @DiffIgnore
    
    //PS:两个注解不可以同时出现在一个类中
    

    如果有没说明白的地方可以留言

  • 相关阅读:
    电脑如何录屏?分享4个屏幕录制的好方法,建议收藏
    Cesium 根据飞机航线计算飞机的Heading(偏航角)、Pitch(俯仰角)、Roll(翻滚角)
    gitlab docker部署,备份,恢复。附踩坑记录
    生态环境影响评价制图流程
    加入 Elastic 贡献者计划的 5 个理由
    【OpenCV 例程200篇】214. 绘制椭圆的参数详解
    基于 Github Actions 自动部署 Hexo 博客
    无涯教程-JavaScript - MROUND函数
    关于远程协作可以分享的有很多,今天单说“定期面对面实现反熵”
    已解决:云原生领域的超时挂载Bug — Kubernetes深度剖析
  • 原文地址:https://www.cnblogs.com/Monkey9527/p/15932558.html