11

我有以下实体;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 Ticketand fetch all WorkOrderinside ticket.workOrders”),通过急切加载执行它们不需要进一步更改,同时指定fetch 需要一个DISTINCT命令

4

2 回答 2

26
  1. 急切加载和获取连接之间存在差异。急切加载并不意味着数据是在同一个查询中加载的。它只是意味着它会立即加载,尽管是通过额外的查询。

  2. 条件总是被转换为 SQL 查询。如果您指定连接,它将是 SQL 中的连接。根据 SQL 的性质,这也会乘以根实体的数据,从而产生您得到的效果。(请注意,您多次获得相同的实例,因此根实体不会在内存中相乘。)

有几种解决方案:

  • 采用distinct(true)
  • 使用不同的根实体转换器 ( .setResultTransformer(Criteria.DISTINCT_ROOT_ENTITY))。
  • 当您不需要按子属性过滤时,请避免加入
  • 当您需要按子属性进行过滤时,请按子查询 ( DetachedCriteria) 进行过滤。
  • 使用batch-size优化N+1问题
于 2012-06-29T06:38:24.017 回答
4

您是否尝试过调用distinct(true)CriteriaQuery?

JPA 2 规范,第 161 页,说:

DISTINCT 关键字用于指定必须从查询结果中消除重复值。

如果未指定 DISTINCT,则不会消除重复值。

javadoc还说:

指定是否消除重复的查询结果。真值将导致消除重复。错误值将导致保留重复项。如果未指定 distinct,则必须保留重复的结果。

急切加载关联时不需要 distinct 的原因可能只是关联没有使用 fetch join 加载,而是使用附加查询。

于 2012-06-14T11:10:17.157 回答