【问题标题】:Hibernate Envers : get only changed fieldsHibernate Envers:仅获取更改的字段
【发布时间】:2017-09-14 08:02:42
【问题描述】:

我怎样才能只从被审计的实体中获取修改的字段?

当我使用时

AuditQuery query = getAuditReader().createQuery().forEntitiesAtRevision(MyEntity.class, revisionNumber).getResultList()

我得到所有领域;但我只想修改字段?

【问题讨论】:

  • "只修改字段",你需要什么格式的字段?字符串数组?在 revisionNumber 和 revisionNumber-1 之间还是当前?请更具体。
  • 我们假设我有实体人员 ID、姓名、出生日期,如果我修改姓名,则创建包含所有字段的修订版,并且我想将姓名从 Melek 更改为 Melek Ellouze。
  • 您希望在重命名之后对名称进行进一步更改吗?还是只是你的重命名?
  • 只有当前版本的更改(仅重命名),但如果对于相同的版本,我更改更改 firstName 和 lastName 我想要更改字段 firstName 从 xxx 到 yyy LastName 从 xxx 到 yyy

标签: java hibernate hibernate-envers revision-history


【解决方案1】:

无修改标志功能

如果您没有在 @Audited 注释上使用 Modified Flags 功能,则获得已审计属性从修订 X 更改为修订 Y 的唯一方法是实际获取两个修订,然后自己比较两个对象实例之间的实际字段值。

具有修改标志功能

假设您在 @Audited 注释上使用 Modified Flags 功能,目前唯一的方法是获取给定实体实例的修订号并使用这些修订和审计的先验知识列,使用 Envers Query API 来询问属性是否针对该修订进行了更改。

显然,这种方法并不理想,因为它确实在用户代码部分强加了一些先验知识,以了解为了获得所需结果而审计的字段。

List<Number> revisions = reader.getRevisions( MyEntity.class, myEntityId );
for ( Number revisionNumber : revisions ) {
  for ( String propertyName : propertyNamesToCheckList ) {
    final Long hits = reader.createQuery()
      .forRevisionsOfEntity( MyEntity.class, false, false  )
      .add( AuditEntity.id().eq( myEntityId ) )
      .add( AuditEntity.revisionNumber().eq( revisionNumber ) )
      .add( AuditEntity.propertyName( propertyName ).hasChanged() )
      .addProjection( AuditEntity.id().count() )
      .getSingleResult();

    if ( hits == 1 ) {
      // propertyName changed at revisionNumber
    }
    else {
      // propertyName didn't change at revisionNumber
    }
  }
}

修改标志属性更改查询

在 Hibernate Envers 6.0 中,我们引入了一个新的查询,它将forRevisionsOfEntity 与修改后的标志查询机制结合起来,不仅可以获得给定实体类类型和主键的修改后的实例,还可以获得一个字段列表每次修订都会修改。

以下伪代码给出了未来 API 的示例:

List results = reader.forRevisionsOfEntityWithChanges( MyEntity.class false )
   .add( AuditEntity.id().eq( entityId ) )
   .getResultList();

Object previousEntity = null;
for ( Object row : results ) {
  Object[] rowArray = (Object[]) row;
  final MyEntity entity = rowArray[0];
  final RevisionType revisionType = (RevisionType) rowArray[2];
  final Set<String> propertiesChanged = (Set<String>) rowArray[3];
  for ( String propertyName : propertiesChanged ) {
    // using the property name here you know
    // 1. that the property changed in this revision (no compare needed)
    // 2. Can get old/new values easily from previousEntity and entity
  }
}

此功能可能会被扩展或更改,因为它将被视为实验性,但这是用户所要求的,我们至少打算基于此功能提供第一次通过在修改过的标志上。

我们目前还没有决定是否或如何支持非修改标志,所以目前唯一的选择是蛮力 bean 比较。

有关此功能的更多详细信息,请参阅HHH-8058

【讨论】:

  • 如果字段发生变化,我找不到返回的方法!我正在使用修改标志功能
  • 我添加了一个示例(未经测试),但它给了你一个想法。
  • @Naros 您对 6.0.0 何时发布有任何估计吗?
  • @A_Di-Matteo 我们现在正在完成最后一点,以最终确定 ORM 的第一次通过 alpha,所以我希望在未来 3-6 个月内的某个时候。
猜你喜欢
  • 2012-12-17
  • 2018-01-04
  • 1970-01-01
  • 1970-01-01
  • 2020-03-11
  • 1970-01-01
  • 1970-01-01
  • 2016-05-27
  • 1970-01-01
相关资源
最近更新 更多