使用 JPA2/Hibernate,我创建了一个实体 A,它具有到实体 X 的单向映射(见下文)。在 A 内部,我还有一个临时成员“t”,我正在尝试使用 @PostLoad 方法进行计算。计算需要访问关联的 X:
@Entity
public class A {
// ...
@Transient
int t;
@OneToMany(orphanRemoval = false, fetch = FetchType.EAGER)
private List listOfX;
@PostLoad
public void calculateT() {
t = 0;
for (X x : listOfX)
t = t + x.someMethod();
}
}
但是,当我尝试加载此实体时,我收到“org.hibernate.LazyInitializationException:非法访问加载集合”错误。
at org.hibernate.collection.AbstractPersistentCollection.initialize(AbstractPersistentCollection.java:363)
at org.hibernate.collection.AbstractPersistentCollection.read(AbstractPersistentCollection.java:108)
at org.hibernate.collection.PersistentBag.get(PersistentBag.java:445)
at java.util.Collections$UnmodifiableList.get(Collections.java:1154)
at mypackage.A.calculateT(A.java:32)
在调试时查看hibernate的代码(AbstractPersistentCollection.java),我发现:
1) 在初始化“listOfX”成员之前调用我的@PostLoad 方法
2) Hibernate 的代码有一个明确的检查,以防止在@PostLoad 期间初始化急切获取的集合:
protected final void initialize(boolean writing) {
if (!initialized) {
if (initializing) {
throw new LazyInitializationException("illegal access to loading collection");
}
throwLazyInitializationExceptionIfNotConnected();
session.initializeCollection(this, writing);
}
}
我想解决这个问题的唯一方法是停止使用 @PostLoad 并将初始化代码移动到 getT() 访问器中,添加一个同步块。但是,我想避免这种情况。
那么,有没有办法在调用 @PostLoad 之前执行急切获取?我不知道有一个 JPA 设施可以做到这一点,所以我希望有一些我不知道的东西。
另外,也许 Hibernate 的专有 API 可以控制这种行为?