11

使用 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 可以控制这种行为?

4

3 回答 3

2

这可能为时已晚,但 hibernate 似乎不支持默认的 jpa fetchtype 选项

@OneToMany(orphanRemoval = false, fetch = FetchType.EAGER)

您必须使用休眠特定的一个:

@LazyCollection(LazyCollectionOption.FALSE)
于 2014-02-19T05:46:27.993 回答
0

我不知道如何解决这个问题,但我认为一些重构可能会有所帮助,想法是将代码移动到@PostConstruct

因此,例如,您的课程将是:

@Entity  
public class A {  
    // ...
    @Transient
    int t;

@OneToMany(orphanRemoval = false, fetch = FetchType.EAGER)  
private List listOfX;  

@PostConstruct
public void calculateT() {
    t = 0;
    for (X x : listOfX)
        t = t + x.someMethod();
}

}

服务器将在完成 bean 的所有容器服务初始化后立即调用 PostConstruct。

于 2010-12-22T09:37:28.287 回答
0

更新了错误报告的链接:

https://hibernate.atlassian.net/browse/HHH-6043

这在 4.1.8 和 4.3.0 或更高版本中已修复

于 2019-04-26T10:45:12.190 回答