Динамическая функция date_trunc в PostgreSQL округляется точно до заданной отметки времени - PullRequest
0 голосов
/ 04 января 2019

Я использую встроенную PostgreSQL-функцию date_trunc () для агрегации временных окон следующим образом:

SELECT max(time) as time, 
COUNT(*) as ticks 
FROM sensorticks 
WHERE time between (TIMESTAMP'2019-01-04 00:15:00') AND (TIMESTAMP'2019-01-04 14:15:00') 
GROUP BY date_trunc('hour', time)
ORDER BY time desc;

Результат таков:

time                        ticks
2019-01-04 14:14:59         892
2019-01-04 13:59:59         3575
2019-01-04 12:59:59         3552
2019-01-04 11:59:59         3560
2019-01-04 10:59:59         2671

Но я не хочу отрезать первое окно. Мне нужны динамические временные окна с 14:15 и далее (14:15, 13:15 и т. Д.). Это должно выглядеть так:

time                        ticks
2019-01-04 14:14:59         3575
2019-01-04 13:14:59         3575
2019-01-04 12:14:59         3552
2019-01-04 11:14:59         3560
2019-01-04 10:14:59         3575

Как бы я это сделал?

1 Ответ

0 голосов
/ 04 января 2019

Похоже, что временное окно для вашего запроса зависит от верхнего диапазона, указанного в предложении WHERE. Поскольку прошло 15 минут после часа, вы можете попытаться агрегировать со смещением времени на 45 минут:

SELECT
    MAX(time) AS time, 
    COUNT(*) AS ticks 
FROM sensorticks 
WHERE time BETWEEN (TIMESTAMP'2019-01-04 00:15:00') AND (TIMESTAMP'2019-01-04 14:15:00') 
GROUP BY date_trunc('hour', time + interval '45 minutes')
ORDER BY time DESC;

Но в приведенном выше запросе все еще есть проблема, поскольку отображаемое время все еще может быть неправильным. Кроме того, некоторые временные интервалы могут отсутствовать, но вы все равно можете сообщить о них. Чтобы это исправить, мы можем попробовать использовать таблицу календаря:

WITH calendar AS (
    SELECT TIMESTAMP '2019-01-04 14:00:00' AS ts, TIMESTAMP '2019-01-04 14:14:59' AS display UNION ALL
    SELECT TIMESTAMP '2019-01-04 13:00:00', TIMESTAMP '2019-01-04 13:14:59' UNION ALL
    SELECT TIMESTAMP '2019-01-04 12:00:00', TIMESTAMP '2019-01-04 12:14:59' UNION ALL
    SELECT TIMESTAMP '2019-01-04 11:00:00', TIMESTAMP '2019-01-04 11:14:59' UNION ALL
    SELECT TIMESTAMP '2019-01-04 10:00:00', TIMESTAMP '2019-01-04 10:14:59'
)

Затем мы можем присоединиться к этой календарной таблице, используя оригинальный запрос:

SELECT
    c.display AS time, 
    COUNT(*) AS ticks 
FROM calendar c
LEFT JOIN sensorticks s
    ON c.ts = date_trunc('hour', s.time + interval '45 minutes')
WHERE s.time BETWEEN (TIMESTAMP'2019-01-04 00:15:00') AND (TIMESTAMP'2019-01-04 14:15:00') 
GROUP BY c.display
ORDER BY c.display DESC;
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...