我正在尝试创建一个方法,该方法将返回任意长度的日期系列,然后可以使用该方法生成按小时/天/月分组的使用情况报告。
这可以针对多个数据库非常重要。我有一个单独实现的解决方案,但它相当笨拙。
这是用例:
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
很公平。
然而,即使在查看和玩dateAdd
和timestampAdd
后,我也无法找到一个有效的解决方案,更不用说简洁明了了。
标准化区间运算的最简单方法是使用DSL.timestampAdd()
,它与TIMESTAMP
数据类型(即TIMESTAMP WITHOUT TIME ZONE
)一起使用。支持DSL.offsetDateTimeAdd()
is on the roadmap,从jOOQ 3.11开始尚未提供。
话虽如此,标准化OffsetDateTime
将导致其自身的困难,因为各种RDBMS对TIMESTAMP WITH TIME ZONE
的含义有不同的理解。