我有以下实体;Ticket 包含一组 0,N 个 WorkOrder:
@Entity
public class Ticket {
...
@OneToMany(mappedBy="ticket", cascade = CascadeType.ALL, fetch = FetchType.LAZY)
private List<WorkOrder> workOrders = null;
...
}
@Entity
public class WorkOrder {
...
@ManyToOne
@JoinColumn(nullable = false)
private Ticket ticket;
}
我正在加载门票并获取属性。所有 0,1 属性都没有问题。对于 workOrders,我使用此答案获取以下代码。
CriteriaBuilder criteriaBuilder = this.entityManager.getCriteriaBuilder();
CriteriaQuery<Ticket> criteriaQuery = criteriaBuilder
.createQuery(Ticket.class);
Root<Ticket> rootTicket = criteriaQuery.from(Ticket.class);
ListAttribute<? super Ticket, WorkOrder> workOrders =
rootTicket.getModel().getList("workOrders", WorkOrder.class);
rootTicket.fetch(workOrders, JoinType.LEFT);
// WHERE logic
...
criteriaQuery.select(rootTicket);
TypedQuery<Ticket> query = this.entityManager.createQuery(criteriaQuery);
return query.getResultList();
结果是,在一个应该返回我 1 个工单和 5 个工作订单的查询中,我检索了同一个工单 5 次。
如果我只是将 workOrders 设为 Eager Fetch 并删除 fetch 代码,它就可以正常工作。
谁能帮我?提前致谢。
更新:
关于为什么我对 JB Nizet 的回答不满意的一种解释(即使最终它有效)。
当我只是使关系急切时,JPA 正在检查与我使它变得懒惰并将 fetch 子句添加到 Criteria / JPQL 时完全相同的数据。ListAttribute
正如我为 Criteria 查询定义的那样,各种元素之间的关系也很清楚。
对于JPA在两种情况下都没有返回相同数据的原因,有一些合理的解释吗?
更新赏金:虽然 JB Nizet 的回答确实解决了这个问题,但我仍然觉得这毫无意义,给定两个具有相同含义的操作(“Get Ticket
and fetch all WorkOrder
inside ticket.workOrders
”),通过急切加载执行它们不需要进一步更改,同时指定fetch 需要一个DISTINCT
命令