一尘不染

忽略关系中的FetchType.EAGER

hibernate

在大型应用程序中,我与EAGER的关系存在问题。此应用程序中的某些实体EAGER与其他实体具有关联。这在某些功能上成为“毒药”。

现在,我的团队需要优化此功能,但是 我们无法将访存类型更改为LAZY ,因为我们需要重构整个应用程序。

所以,我的问题是:是否有一种方法可以执行特定的查询而忽略我返回的实体中的EAGER关联?

示例:当我有这个实体Person时,我在查询以查找Person时不带地址列表。

@Entity
public class Person {

  @Column
  private String name;

  @OneToMany(fetch=FetchType.EAGER)
  private List<String> address;

}

Query query = EntityManager.createQuery("FROM Person person");
//list of person without the address list! But how???
List<Person> resultList = query.getResultList();

谢谢!

更新

我发现的唯一方法是不返回实体,仅返回实体的某些字段。但是我想找到一个可以返回实体(在我的示例中为Person实体)的解决方案。

我在想是否有可能在Hibernate中两次映射同一张表。这样,我可以在没有EAGER关联的情况下映射同一张表。这在某些情况下会帮助我…


阅读 384

收藏
2020-06-20

共1个答案

一尘不染

这些年来,在Hibernate上 尚无法
覆盖EAGER映射。从最新的Hibernate文档(5.3.10.Final):

尽管JPA标准指定您可以在运行时使用javax.persistence.fetchgraph提示覆盖EAGER提取关联,但是当前Hibernate并未实现此功能,
因此无法懒惰地提取EAGER关联 。有关更多信息,请查看
HHH-8776 Jira问题。

当执行JPQL查询时,如果省略EAGER关联,则Hibernate将为需要急切获取的每个关联发出辅助选择,这可能导致dto N + 1查询问题。

因此,最好使用LAZY关联,并且仅在每个查询的基础上热切地获取它们。

和:

EAGER的获取策略 不能在每个查询的基础上被覆盖
,因此即使您不需要该关联,也总是可以对其进行检索。此外,如果您忘记在JPQL查询中加入FETCH
EAGER关联,Hibernate将使用辅助语句对其进行初始化,这反过来可能导致N + 1查询问题。

2020-06-20