我们有以下实体层次结构和EntityListener
@Entity
@EntityListeners(value = DummyListener.class)
@Audited
public class A {
@OneToMany(cascade = CascadeType.ALL, mappedBy = "a", fetch = FetchType.LAZY, orphanRemoval = true)
@Fetch(FetchMode.SELECT)
private Set<B> bs = new LinkedHashSet<B>(0);
}
@Entity
@Audited
public class B {
@ManyToOne(fetch = FetchType.EAGER, cascade = { CascadeType.MERGE, CascadeType.PERSIST })
@JoinColumn(name = "a_id", nullable = false)
private A a;
}
public class DummyListener {
@PreUpdate
public void beforeUpdate(A entity) {
entity.set...(...);
...
B old = entity.getBs().iterator().next();
old.set...(...);
...
B b = new B();
b.setA(entity);
entity.getBs().add(b);
...
}
}
在实体监听器中,我们想要将新的B实体添加到A实体集合中。但是在保存A之后,envers没有为B审计表中的新B实例创建审计记录。 除新B实例外,其他每个更改都有审计记录。
我们错了什么?可以用这种方式创建B的新实例吗?
答案 0 :(得分:1)
根据JPA规范2.1:
通常,可移植应用程序的生命周期方法不应调用EntityManager或查询操作,访问其他实体实例或修改同一持久化上下文中的关系。
这里的要点是,添加新的B
并将其与A
相关联意味着您正在修改实体之间的关系,尽管是新实体B
。
所以采取以下代码:
@PreUpdate
public void onPreUpdate(Object object) {
if ( object instanceof A ) {
final A a = (A) object;
final B b = new B();
b.setA( a );
a.getBs().add( b );
}
}
如果仅使用Hibernate进行测试并且没有审核,则会注意到B
的新实例在@PreUpdate
的生命周期回调中创建时甚至不会保留。
因此,如果Hibernate不允许这样的用例,那么Envers甚至无法检测状态变化并执行必要的收集工作单元。