【问题标题】:select greatest element per group in querydsl在querydsl中选择每组最大的元素
【发布时间】:2020-02-09 04:31:54
【问题描述】:

我有一个大致如下的 sql 表:

+-----+------+-------+
|  id | type | value |
+-----+------+-------+
|  1  |   X  |   20  |
|  2  |   X  |   30  |
|  3  |   Y  |  200  |
|  4  |   Y  |  500  |
|  5  |   Y  |  300  |
|  6  |   Z  |    5  |
+-----+------+-------+

对于每种类型,我想检索具有最大值的行。这是我的预期结果:

+-----+------+
|  id | type |
+-----+------+
|  2  |   X  | <-- had value = 30
|  4  |   Y  | <-- had value = 500
|  6  |   Z  | <-- had value = 5
+-----+------+

在 SQL 中,这可以表示如下(假设每种类型没有两个具有相同值的条目,我可以排除):

select t1.id, t1.type from T t1
inner join (
  select t2.type, max(t2.value) as max_value from T t2
  group by t2.type
) on t1.type = t2.type
  and t1.value = max_value

但是我找不到使用 QueryDSL(第 4 版)表达相同内容的方法。我试过这个:

final JPQLQuery<Tuple> subquery = JPAExpressions
    .from(q2)
    .select(q2.type, q2.value.max())
    .groupBy(q2.type);
final JPQLQuery<Tuple> query = JPAExpressions
    .from(q1)
    .select(q1.id, q1.type)
    .from(q1)
    .innerJoin(subquery) // <-- not allowed
    .on(q1.type.eq(q2.type), q1.value.eq(q2.value.max()));

但是innerJoin()(和其他连接方法)只接受一个表达式作为参数,而不是另一个查询。 from() 也是如此。

【问题讨论】:

    标签: java sql jpa greatest-n-per-group querydsl


    【解决方案1】:

    子查询可以以exists-表达式的形式放入外部查询的where子句中:

    final JPQLQuery<Tuple> subquery = JPAExpressions
        .from(q2)
        .select(q2.type, q2.value.max())
        .groupBy(q2.type);
    final JPQLQuery<Tuple> query = JPAExpressions
        .from(q1)
        .select(q1.id, q1.type)
        .from(q1)
        .where(subquery
            .having(q1.type.eq(q2.type), q1.value.eq(q2.value.max()))
            .exists());
    

    请注意,此查询可能效率很低。

    【讨论】:

    • 因为查询对我来说确实很慢,所以我只是按降序排序,然后跳过代码中键入的重复项。这对我来说是可行的,因为每个组最多只包含几个条目
    猜你喜欢
    • 2018-10-27
    • 2011-05-29
    • 2021-11-22
    • 2017-04-08
    • 2021-10-11
    • 2014-08-24
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多