Hibernate Envers:只获取更改的字段

Hibernate Envers : get only changed fields

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

当我使用

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

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

没有修改标志功能

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

具有修改标志功能

假设您在 @Audited 注释上使用 Modified Flags 功能,目前唯一的方法是获取给定实体实例的修订号并使用它们审计列的修订和先验知识,使用 Envers 查询 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 与修改后的标志查询机制相结合,不仅可以获取给定实体 class 类型和主键的修改实例,还有每次修订时修改的字段列表。

下面的伪代码给出了一个未来的例子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