繁体   English   中英

无法获取连接延迟初始化 collections

[英]Can't fetch join lazy initialized collections

由于 MultipleBagFetchException,我无法执行简单的 fetchjoin。

@Entity
public class Person {

    @OneToMany(mappedBy="person",fetch=FetchType.LAZY)
    private List<Auto> autos;    
}


@Entity
public class Auto {

    @ManyToOne
    @JoinColumn(name = "person_id", nullable = false)
    private Person person;

   @OneToMany(mappedBy="auto",fetch=FetchType.LAZY)
   private List<Tool> tools;

}

@Entity
@Table(name="tool")
public class Tool {

    @ManyToOne
    @JoinColumn(name = "auto_id", nullable = false)
    private Auto auto;
}

如您所见,我所有的关联都使用默认的 fetchtype。

 @Query("SELECT p FROM Person p JOIN FETCH p.autos a JOIN FETCH a.tools")
 List<Person>findAll();

结果:

Caused by: org.hibernate.loader.MultipleBagFetchException: cannot simultaneously fetch multiple bags: [com.example.entities.Person.autos, com.example.entities.Auto.tools]

我已阅读有关此异常的信息,但在这些情况下,此异常的原因是 collections 使用了 EAGER 提取类型。 那这个呢? 这是最简单的实体关系。

最重要的是,假设我们不允许触摸实体。 如何仅在查询端解决这个问题?

有一种方法可以在不触及实体的情况下避免 n+1 次查询,只需更改 findAll 的查询即可。 我们可以编写一个包装器 function,它将首先加载带有汽车的人员,然后他们在单个 select 中获取所有工具。

个人资料库

    @Query("SELECT distinct p FROM Person p JOIN FETCH p.autos a")
    List<Person> findAll();

包装代码

    List<Person> persons = personRepository.findAll();
    Session session = (Session) entityManager.getDelegate();
    List<Auto> autos = new ArrayList<>();
    for (Person person : persons) {
        if(!CollectionUtils.isEmpty(person.getAutos())) {
            autos.addAll(person.getAutos());
        }
    }
    try{
        autos = session.createQuery("select distinct a from Auto a Join fetch a.tools " +
                " where a in :autos", Auto.class)
                .setParameter("autos", autos)
                .setHint(QueryHints.PASS_DISTINCT_THROUGH, false)
                .getResultList();
    } catch (Exception ex) {
        ex.printStackTrace();
    }

第一个查询将是:

SELECT DISTINCT
          person0_.id       AS id1_6_0_,
          autos1_.id        AS id1_0_1_,
          person0_.name     AS name2_6_0_,
          autos1_.name      AS name2_0_1_,
          autos1_.person_id AS person_i3_0_1_,
          autos1_.person_id AS person_i3_0_0__,
          autos1_.id        AS id1_0_0__
FROM
          Person person0_
          INNER JOIN
                    Auto autos1_
          ON
                    person0_.id=autos1_.person_id

生成的第二个查询将是:

    SELECT
          auto0_.id        AS id1_0_0_,
          tools1_.id       AS id1_8_1_,
          auto0_.name      AS name2_0_0_,
          auto0_.person_id AS person_i3_0_0_,
          tools1_.auto_id  AS auto_id3_8_1_,
          tools1_.name     AS name2_8_1_,
          tools1_.auto_id  AS auto_id3_8_0__,
          tools1_.id       AS id1_8_0__
FROM
          Auto auto0_
          INNER JOIN
                    Tool tools1_
          ON
                    auto0_.id=tools1_.auto_id
WHERE
          auto0_.id IN (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)

除此之外,我相信我们的选项是有限的,我们将不得不更改工具实体 FetchMode 或为默认 FetchMode.SELECT 添加 BatchSize 以便在单独的查询中获取工具。

    @OneToMany(mappedBy = "auto", fetch = FetchType.LAZY)
    @Fetch(FetchMode.SUBSELECT)
    private List<Tool> tools;

查询将是

SELECT
          tools0_.auto_id AS auto_id3_8_1_
        , tools0_.id      AS id1_8_1_
        , tools0_.id      AS id1_8_0_
        , tools0_.auto_id AS auto_id3_8_0_
        , tools0_.name    AS name2_8_0_
FROM
          Tool tools0_
WHERE
          tools0_.auto_id IN
          (
                    SELECT
                              autos1_.id
                    FROM
                              Person person0_
                              INNER JOIN
                                        Auto autos1_
                              ON
                                        person0_.id=autos1_.person_id
          )

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM