Hibernate 不允许获取多个包,因为这会生成笛卡尔积,并且对于在 Hibernate 术语中称为baggs的无序列表,即使基础集合没有那些重复的行,这也会导致重复条目。因此,Hibernate 只是在编译 JPQL 查询时阻止了这种情况。
现在,您会发现很多答案、博客文章、视频或其他资源告诉您在收藏中使用 aSet
而不是 a List
。
这是可怕的建议。不要那样做!
使用Sets
而不是Lists
将使MultipleBagFetchException
消失,但笛卡尔积仍然存在。
正确的修复
JOIN FETCH
而不是在单个 JPQL 或 Criteria API 查询中使用多个:
List<Post> posts = entityManager
.createQuery(
"select p " +
"from Post p " +
"left join fetch p.comments " +
"left join fetch p.tags " +
"where p.id between :minId and :maxId", Post.class)
.setParameter("minId", 1L)
.setParameter("maxId", 50L)
.getResultList();
您可以执行以下技巧:
List<Post> posts = entityManager
.createQuery(
"select distinct p " +
"from Post p " +
"left join fetch p.comments " +
"where p.id between :minId and :maxId ", Post.class)
.setParameter("minId", 1L)
.setParameter("maxId", 50L)
.setHint(QueryHints.PASS_DISTINCT_THROUGH, false)
.getResultList();
posts = entityManager
.createQuery(
"select distinct p " +
"from Post p " +
"left join fetch p.tags t " +
"where p in :posts ", Post.class)
.setParameter("posts", posts)
.setHint(QueryHints.PASS_DISTINCT_THROUGH, false)
.getResultList();
只要您使用 最多获取一个集合JOIN FETCH
,就可以了。通过使用多个查询,您将避免笛卡尔积,因为任何其他集合,但第一个集合是使用辅助查询获取的。