我有多个通过JPA2 Criteria Query查询的实体。
我可以加入其中两个实体并立即获得结果:
CriteriaBuilder criteriaBuilder = em.getCriteriaBuilder();
CriteriaQuery<LadungRgvorschlag> criteriaQuery = criteriaBuilder.createQuery(LadungRgvorschlag.class);
Root<LadungRgvorschlag> from = criteriaQuery.from(LadungRgvorschlag.class);
Join<Object, Object> ladung = from.join("ladung");
from.fetch("ladung", JoinType.INNER);
然后我尝试加入一个像这样的附加表:
ladung.join("ladBerechnet");
ladung.fetch("ladBerechnet", JoinType.LEFT);
我收到以下错误:
org.hibernate.QueryException: query specified join fetching, but the owner of the fetched association was not present in the select list [FromElement{explicit,not a collection join,fetch join,fetch non-lazy properties,classAlias=generatedAlias3,role=null,tableName=ladberechnet,tableAlias=ladberechn3_,origin=ladungen ladung1_,columns={ladung1_.id ,className=de.schuechen.beans.tms.master.LadBerechnet}}] [select generatedAlias0 from de.schuechen.beans.tms.master.LadungRgvorschlag as generatedAlias0 inner join generatedAlias0.ladung as generatedAlias1 inner join generatedAlias1.ladBerechnet as generatedAlias2 left join fetch generatedAlias1.ladBerechnet as generatedAlias3 inner join fetch generatedAlias0.ladung as generatedAlias4 where ( generatedAlias0.erledigt is null ) and ( generatedAlias0.belegart in (:param0, :param1) ) and ( generatedAlias1.fzadresse in (:param2, :param3) ) and ( generatedAlias1.zudatum<=:param4 ) and ( 1=1 ) order by generatedAlias0.belegart asc, generatedAlias1.fzadresse asc, generatedAlias1.zudatum asc, generatedAlias1.zulkw asc]
我如何告诉JPA / Hibernate,它应该一次选择所有实体?
答案 0 :(得分:11)
使用 JPA 'JPA的一些方言'你可以链接连接提取,但我认为你不能/应该同时进行连接和连接提取。
例如,如果我们的Program
与Reward
的{{1}}具有一对多的关系,并且与Duration
有关系,则以下JPQL将获得特定的预先获得奖励和持续时间的实例:
SELECT DISTINCT
program
FROM
Program _program
LEFT JOIN FETCH
_program.rewards _reward
LEFT JOIN FETCH
_reward.duration _duration
WHERE
_program.id = :programId
}
使用等效的Criteria代码:
CriteriaBuilder criteriaBuilder = entityManager.getCriteriaBuilder();
CriteriaQuery<Program> criteriaQuery = criteriaBuilder.createQuery(Program.class);
Root<Program> root = criteriaQuery.from(Program.class);
Fetch<Program, Reward> reward = root.fetch("rewards", JoinType.LEFT);
Fetch<Reward, Duration> duration = reward.fetch("duration", JoinType.LEFT);
criteriaQuery.where(criteriaBuilder.equal(root.get("id"), programId));
TypedQuery<program> query = entityManager.createQuery(criteriaQuery);
return query.getSingleResult();
请注意,此处不需要中间变量奖励和持续时间,但它们仅用于提供信息。 root.fetch("rewards", JoinType.LEFT).fetch("duration", JoinType.LEFT)
会产生同样的效果。
答案 1 :(得分:7)
对于JPA来说,你不能在Criteria API查询中链接连接提取(来自规范的引用):
fetch方法引用的关联或属性必须是 从实体引用或作为结果返回的embeddable引用 的查询。获取连接具有与之相同的连接语义 相应的内部或外部连接,相关对象除外 不是查询结果中的顶级对象,也不能被引用 其他地方的查询。
JPQL查询也不支持它:
FETCH JOIN子句右侧引用的关联 必须是引用的关联或元素集合 作为查询结果返回的实体或嵌入式。
不允许为其指定标识变量 FETCH JOIN子句右侧引用的对象,和 因此,对隐式获取的实体或元素的引用不能 出现在查询的其他地方。
使用HQL似乎是可能的:Hibernate documentation EclipseLink没有提供这样的扩展,因此Hibernate接受了以下查询的语法,但EclipseLink不接受:
SELECT a FROM A a LEFT JOIN FETCH a.bb b LEFT JOIN FETCH b.cc
在EclipseLink中,可以通过query hints完成同样的工作。