带有元组条件的QueryDSL和SubQuery

时间:2015-12-02 15:07:06

标签: java hibernate jpa syntax querydsl

我正在尝试在QueryDSL中编写一个查询,以获取按其parentId分组的表中最旧的元素。

SQL等价物应该是:

SELECT a.* FROM child a
    INNER JOIN
    (
        SELECT parentId, MAX(revision) FROM child GROUP BY parentId
    ) b
    ON ( 
        a.parentId = b.parentId AND a.revision = b.revision
    )

现在在QueryDSL中,我坚持使用语法。

JPQLQuery<Tuple> subquery = JPAExpressions
                .select(child.parent, child.revision.max())
                .from(child)
                .groupBy(child.parent);

HibernateQuery<Child> query = new HibernateQuery<>(session);
query.from(child)
    .where(child.parent.eq(subquery.???).and(child.revision.eq(subquery.???))));

如何使用子查询编写此查询?

表格如下所示:

___parent___ (not used in this query, but exists)
parentId
P1       | *
P2       | *
P3       | *

___child___
parentId | revision
P1       | 1       | *
P1       | 2       | *
P1       | 3       | *
P2       | 2       | *
P2       | 3       | *
P3       | 1       | *

___result from child, the highest revision for each parentId___
P1       | 3       | *
P2       | 3       | *
P3       | 1       | *

到目前为止我尝试了什么:

.where(JPAExpressions.select(child.parent,child.revision).eq(subquery));

-> org.hibernate.hql.internal.ast.QuerySyntaxException: unexpected end of subtree

和许多语法错误...

我现在使用脏循环,因为我还没有找到解决方案。

3 个答案:

答案 0 :(得分:1)

在JPA中,子查询只能出现在where部分中。

以下是我对您的查询的看法

select(child).from(child).where(child.revision.eq(
  select(child2.revision.max())
 .from(child2)
 .where(child2.parent.eq(child.parent))
 .groupBy(child2.parent))).fetch()

答案 1 :(得分:1)

您可以使用Expressions.list()为in子句指定多个列:

query.from(child).where(Expressions.list(child.parent, child.revision).in(subquery));

另一种方法是使用innerJoin(),就像在原始SQL中一样。

答案 2 :(得分:0)

Expressions.list(ENTITY.year, ENTITY.week).in(//
                    Expressions.list(Expressions.constant(1029), Expressions.constant(1)),
                    Expressions.list(Expressions.constant(1030), Expressions.constant(1)),
                    Expressions.list(Expressions.constant(1031), Expressions.constant(1))

将是您想要的,但是QueryDSL会从中生成错误的SQL:

((p0_.year , p0_.week) in (1029 , 1 , (1030 , 1) , (1031 , 1)))