JPA2 Criteria查询实体层次结构

时间:2010-09-28 13:06:20

标签: jpa-2.0

假设我有以下实体域:

@Entity
@Inheritance(strategy=InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name="TYPE")
public abstract class Entity1 {
//some attributes
}

@Entity 
@DiscriminatorValue("T1")
public class Entity2 extends Entity1 {
    @OneToMany(fetch=FetchType.EAGER, cascade = { CascadeType.ALL }, mappedBy="parent")
    @Cascade(org.hibernate.annotations.CascadeType.DELETE_ORPHAN)
    private Set<Entity1Detail> details = new HashSet<Entity1Detail>();
}

@Entity
public class Entity1Detail {
    @ManyToOne
    @JoinColumn(name="REF")
    private Entity2 parent;

    @Basic
    private Integer quantity;
}

@Entity
@DiscriminatorValue("T2")
public class Entity3 extends Entity1 {
//some other attributes
}

当我执行JPQL查询时:

select e from Entity1 e left join e.details d where d.quantity > 1

运行良好(左连接; P)。但是,当我尝试使用JPA2 Criteria API构建相同的查询时:

CriteriaBuilder builder = em.getCriteriaBuilder();
CriteriaQuery q = builder.createQuery();
Root r = q.from(Entity1.class);
q.select(r);
q.where(builder.gt(r.join("details", JoinType.LEFT).get("quantity"), 1));

我在“join”中获得NPE,因为属性“details”不属于Entity1(实际上是真的,我必须在Entity2.class上选择)。问题是,当我必须使用Criteria API构建我的动态查询时,我对层次结构一无所知,我只是传递了一个类。

我知道Criteria API是类型安全的,所有这一切,但是有办法解决这个问题吗?使用别名(如前所述,我使用Hibernate Criteria API,遍历带别名的连接):

Criteria c = session.createCriteria(Entity1.class);
c.createAlias("details", "d");
c.add(Restrictions.ge("d.quantity", 1));

1 个答案:

答案 0 :(得分:2)

您需要将查询基于entity2.details。由于条件API是类型安全的,因此它捕获了entity1没有名为“details”的字段

CriteriaBuilder builder = em.getCriteriaBuilder();
CriteriaQuery q = builder.createQuery();
Root r = q.from(Entity2.class);    // Must use subclass as root
q.select(r);
q.where(builder.gt(r.join("details", JoinType.LEFT).get("quantity"), 1));

由于Entity2扩展了Entity1,因此您可以安全地将结果转换为父类型。例如:

CriteriaQuery<Entity1> q = builder.createQuery(Entity1.class);
Root r = q.from(Entity2.class);    // Must use subclass as root

将返回Entity1的列表

CriteriaQuery<Entity2> q = builder.createQuery(Entity2.class);
Root r = q.from(Entity2.class);    // Must use subclass as root

将返回Entity2的列表

编辑:

我想我误解了这里的目标。如果你想要所有Entity1 UNLESS它们是Entity2并且details.quantity&lt; = 1,你需要做更多。

您不能使用从Entity1Detail到Entity1的左连接,因为这不是严格类型安全的。相反,您需要以某种方式将Entity2加入Entity1Detail。可能在这里使用的最佳工具是相关子查询

CriteriaQuery<Entity1> q = builder.createQuery(Entity1.class);
Root<Entity1> ent1 = q.from(Entity1.class);

SubQuery<Entity2> subq = q.subquery(Entity2.class);
Root<Entity2> ent2 = subq.from(Entity2.class);
Path<Integer> quantity = ent2.join("details", JoinType.LEFT).get("quantity");
Predicate lessThan = builder.lte(quantity,1);
Predicate correlatedSubqJoin = cb.equal(ent1,ent2)
subq.where(lessThan, correlatedSubqJoin);

q.select(ent1);
q.where(builder.exists(subq).not());

条件API不知道您是单表继承,因此您必须为所有继承策略编写查询,包括加入的继承策略。

相关问题