使用JOOQ创建交叉目标日期系列

问题描述 投票:0回答:1

我正在尝试创建一个方法,该方法将返回任意长度的日期系列,然后可以使用该方法生成按小时/天/月分组的使用情况报告。

这可以针对多个数据库非常重要。我有一个单独实现的解决方案,但它相当笨拙。

这是用例:


Table<Record> generateDateSeries(OffsetDateTime minDate, OffsetDateTime maxDate) {
  ## TODO implement 
} 
Table<Record> truncatedDateSeries = generateDates(OffsetDateTime.now().minus(Duration.ofDays(7)), OffsetDateTime.now());

// So that it can then be used in a query such as the following

/*
 * Get all of the usage with a date range grouped by a date part
 */
db.select(
         truncatedDateSeries.field(columnDate).cast(OffsetDateTime.class),
         dataTable.field(columnDownload).cast(BigDecimal.class))
  .from(truncatedDateSeries)
  .leftOuterJoin(dataTable)
  .using(field(columnDate)).fetch()

我目前对generateDateSeries的实施如下:

Table<Record> generateDates(OffsetDateTime minDate, OffsetDateTime maxDate) {
    Name cte = name("dateSeries");
    Name dateValue = name("dateValue");

    return DSL.withRecursive(cte).as(
      select(value(minDate).as(field(dateValue)))
        .unionAll(
          select(
            field(dateValue, OffsetDateTime.class).add(1)).from(cte).where(
            field(dateValue).add(1).lessThan(maxDate)
          )
        )
    ).select().from(cte).asTable();
  }

最终,当我们最终运行查询时,我们得到了

ERROR: operator does not exist: timestamp with time zone + integer
SQL state: 42883
Hint: No operator matches the given name and argument type(s). You might need to add explicit type casts.
Character: 188

很公平。

然而,即使在查看和玩dateAddtimestampAdd后,我也无法找到一个有效的解决方案,更不用说简洁明了了。

java sql jooq
1个回答
0
投票

标准化区间运算的最简单方法是使用DSL.timestampAdd(),它与TIMESTAMP数据类型(即TIMESTAMP WITHOUT TIME ZONE)一起使用。支持DSL.offsetDateTimeAdd() is on the roadmap,从jOOQ 3.11开始尚未提供。

话虽如此,标准化OffsetDateTime将导致其自身的困难,因为各种RDBMS对TIMESTAMP WITH TIME ZONE的含义有不同的理解。

© www.soinside.com 2019 - 2024. All rights reserved.