gpt4 book ai didi

java - 对象不同 hasChanges,其中不应检测到任何更改

转载 作者:行者123 更新时间:2023-11-30 10:26:42 26 4
gpt4 key购买 nike

我正在使用 java-object-diff获取 JAXB 从 xml 解析的两个对象之间的差异。在下面的示例中,我使用相同的字符串来测试是否没有差异,但是 log.info("has changes: " + diff5.hasChanges());日志 true .

JAXBContext context1 = JAXBContext.newInstance(Item.class);
Unmarshaller m1 = context1.createUnmarshaller();
Item base = (Item) m1.unmarshal(new StringReader(s));
Item working = (Item) m1.unmarshal(new StringReader(s));
DiffNode diff5 = ObjectDifferBuilder
.buildDefault()
.compare(working, base);
log.info("has changes: " + diff5.hasChanges());
diff5.visit((node, visit) -> {
final Object baseValue = node.canonicalGet(base);
final Object workingValue = node.canonicalGet(working);
final String message = node.getPath() + " changed from " +
baseValue + " to " + workingValue;
System.out.println(message);
});

我从 System.out.println 得到的消息总是一样的,说它已经从 null 改变了。至 <the actual value> 每个 属性都会发生这种情况。例如

content changed from null to Mit dem Wasserinonisator

我已经验证了这两个 Items具有相同的内容,两者都不是 null ,但内容完全相同。

Item是一个有很多子类的 pojo(所有的 getter 和 setter 都存在),例如

public class Item {

@XmlElement(name = "ASIN", required = true)
protected String asin;
@XmlElement(name = "ParentASIN")
protected String parentASIN;
@XmlElement(name = "Errors")
protected Errors errors;
@XmlElement(name = "DetailPageURL")
protected String detailPageURL;
@XmlElement(name = "ItemLinks")
protected ItemLinks itemLinks;
@XmlElement(name = "SalesRank")
protected String salesRank;
@XmlElement(name = "SmallImage")
protected Image smallImage;
}

有没有办法让 java-object-diff 工作,让它正确比较值?

最佳答案

仔细查看your code之后我知道出了什么问题。第一个问题是 JAXB 不会生成 equals 方法。在大多数情况下,这不是问题,因为 ObjectDiffer 可以基于层次结构建立对象之间的关系。当涉及有序或无序的 Collections 时,事情会变得更加复杂,因为 ObjectDiffer 需要某种方式来建立基础和工作实例中的集合项之间的关系。默认情况下,它依赖于底层集合的查找机制(通常涉及一个或多个方法 hashCodeequalscompareTo。)

在你的情况下,这种关系无法建立,因为你的类(尤其是那些包含在 ListsSets 中的类)都没有实现适当的 equals方法。这意味着实例永远只等于它们自己。由于负责的类表示值对象并且没有任何可用于轻松建立关系的硬标识符,这一事实使情况变得更加复杂。因此,唯一的选择是提供自定义的 equals 方法来简单地比较所有属性。结果是,对这些对象进行最细微的更改都会导致 ObjectDiffer 将基本版本标记为 REMOVED 并将工作版本标记为 ADDED。但它也不会将它们标记为 CHANGED,因为它们实际上并没有改变。就是这样。

我不确定让 JAXB 生成自定义 equals 方法有多容易,所以这里有一些可能使用 java-object-diff 的替代解决方案:

  1. 为有问题的类型实现您自己的 de.danielbechler.diff.identity.IdentityStrategy 并将它们提供给 ObjectDifferBuilder,就像这样(示例使用 Java 8 lambda 斯):

    ObjectDifferBuilder
    .startBuilding()
    .identity()
    .ofCollectionItems(ItemLinks.class, "itemLink").via((working, base) -> {
    ItemLink workingItemLink = (ItemLink) working;
    ItemLink baseItemLink = (ItemLink) base;
    return StringUtils.equals(workingItemLink.getDescription(), baseItemLink.getDescription())
    && StringUtils.equals(workingItemLink.getURL(), baseItemLink.getURL());
    })
    // ...
    .and().build();
  2. 在比较过程中忽略有问题的属性。显然这可能不是您想要的,但如果您并不真正关心特定对象,这是一个简单的解决方案。

    ObjectDifferBuilder
    .startBuilding()
    .inclusion()
    .exclude().type(Item.ImageSets.class)
    .and().build();

导致 JAXB 生成自定义 equals 方法的解决方案将是我的首选方式。我找到了 another post声称这是可能的,所以也许您想先尝试一下,这样您就不必自定义 ObjectDiffer

希望对您有所帮助!

关于java - 对象不同 hasChanges,其中不应检测到任何更改,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45586006/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com