首頁 > 軟體

java中Javers 比較兩個類的差異

2022-02-25 10:01:42

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(BigDecimal.class,new BigDecimalComparatorWithFixedEquals()).build();
Javers javers = JaversBuilder.javers()
    .registerValue(BigDecimal.class,new BigDecimalComparatorWithFixedEquals())
    .registerValue(BigDecimal.class,new BigDecimalComparatorWithFixedEquals())
    .build();

相關注解

在比較兩個類時,可能需要只比較部分欄位或不比較部分欄位,可以考慮使用以下兩個註解

//在一個類中,在欄位上新增該註解,該類中只比較有註解的欄位
@DiffInclude

//在一個類中,在欄位上新增該註解,該類中忽略比較有註解的欄位
@DiffIgnore

//PS:兩個註解不可以同時出現在一個類中

到此這篇關於java中Javers 比較兩個類的差異的文章就介紹到這了,更多相關Javers 比較兩個類的差異內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com