Postgresql конвертировать запрос для просмотра с общим параметром месяца - PullRequest
0 голосов
/ 09 июня 2018

У меня есть очень большой запрос.

В в этом запросе я должен определить диапазон дат, например, где дата между x и y.Определение диапазона дат - это запрос на выборку внутри другого запроса на выборку.

Я хочу, чтобы этот запрос был преобразован в представление.Таким образом, он группирует все результаты по месяцам, например, где дата между '2000-01-01' и '2000-01-31'.

Есть ли какой-то способ сделать это, например

where date in unique month

, чтобы я мог сделать еще один запрос, который просто

select * from temp_view where (max_start_date, max_end_date) overlaps ('2000-01-01', '2000-01-31')

Трудно объяснить ...

Было бы похоже на ввод всех возможных месяцев взапрос.

Это пример запроса

    data_table
    fruit  | start_date  |  end_date
    APPLE    2000-01-14     2000-01-20
    APPLE  | 2000-01-20  |  2000-02-05
    ORANGE | 2000-01-01  |  2000-02-10

    value_table (there is a reading every day)
   trading_date | fruit  | value
    2000-01-01   APPLE    1
    2000-01-02   APPLE    1.2
    2000-01-03   APPLE    2.2
    ...
    2000-02-15   APPLE   4.4
    2000-02-16   APPLE   5.2
    ...
    2000-01-01   ORANGE    2
    2000-01-02   ORANGE    3
    2000-01-03   ORANGE    1
    ...
    2000-02-15   ORANGE   2.4
    2000-02-16   ORANGE   2.2

Запрос:

    with prequery as (SELECT   value, start_date, end_date
        from data_table
            )     
            SELECT 
                month,
                some_value,
                start_date,
                end_date,
                sum(daily_value)
                FROM  prequery
                INNER JOIN value_table
                on fruit=value_table.fruit  
                and trading_date between start_date and  end_date
                ->[[[and trading_date between '2000-01-01' and '2000-01-31']]]<- dont want this.  Want it to group by months.

Вывод (включает не все значения, а просто показывает, к чему я стремлюсьдля)

Month  |  value  |  start_date   | end_date   | sum   |    
2000-01   APPLE     2000-01-20    2000-02-05    (sum of all values between start and end dates in January for APPLE)
2000-02   APPLE     2000-01-14    2000-01-20    (sum of all values between start and end dates in February for APPLE)

1 Ответ

0 голосов
/ 09 июня 2018

Вы ищете представление, из которого вы можете выбрать первый и последний день месяца?

Этот режим создаст представление months со всеми месяцами с января 1970 года по декабрь 2070 года.

CREATE VIEW months
AS
WITH RECURSIVE cte(i, number_of_year, number_of_month, first_day, last_day)
AS
(
SELECT 1 i,
       0 / 12 + 1970 number_of_year,
       0 % 12 + 1 number_of_month,
       (lpad((0 / 12 + 1970)::text, 4, '0') || '-' || lpad((0 % 12 + 1)::text, 2, '0') || '-01')::timestamp first_day,
       (lpad((0 / 12 + 1970)::text, 4, '0') || '-' || lpad((0 % 12 + 1)::text, 2, '0') || '-01')::timestamp + INTERVAL '1 month - 1 day' last_day
UNION ALL
SELECT i + 1 i,
       i / 12 + 1970 number_of_year,
       i % 12 + 1 number_of_month,
       (lpad((i / 12 + 1970)::text, 4, '0') || '-' || lpad((i % 12 + 1)::text, 2, '0') || '-01')::timestamp first_day,
       (lpad((i / 12 + 1970)::text, 4, '0') || '-' || lpad((i % 12 + 1)::text, 2, '0') || '-01')::timestamp + INTERVAL '1 month - 1 day' last_day
       FROM cte
       WHERE i / 12 + 1970 <= 2070
)
SELECT number_of_year,
       number_of_month,
       first_day,
       last_day
       FROM cte;

Но я рекомендую материализовать это для повышения производительности.

CREATE TABLE months
AS
WITH RECURSIVE cte(i, number_of_year, number_of_month, first_day, last_day)
AS
(
SELECT 1 i,
       0 / 12 + 1970 number_of_year,
       0 % 12 + 1 number_of_month,
       (lpad((0 / 12 + 1970)::text, 4, '0') || '-' || lpad((0 % 12 + 1)::text, 2, '0') || '-01')::timestamp first_day,
       (lpad((0 / 12 + 1970)::text, 4, '0') || '-' || lpad((0 % 12 + 1)::text, 2, '0') || '-01')::timestamp + INTERVAL '1 month - 1 day' last_day
UNION ALL
SELECT i + 1 i,
       i / 12 + 1970 number_of_year,
       i % 12 + 1 number_of_month,
       (lpad((i / 12 + 1970)::text, 4, '0') || '-' || lpad((i % 12 + 1)::text, 2, '0') || '-01')::timestamp first_day,
       (lpad((i / 12 + 1970)::text, 4, '0') || '-' || lpad((i % 12 + 1)::text, 2, '0') || '-01')::timestamp + INTERVAL '1 month - 1 day' last_day
       FROM cte
       WHERE i / 12 + 1970 <= 2070
)
SELECT number_of_year,
       number_of_month,
       first_day,
       last_day
       FROM cte;

Я полагаю, вы будете запрашивать это так:

SELECT first_day,
       last_day
       FROM months
       WHERE number_of_year = <year>
             AND number_of_month = <month>;

Затем также создайте следующий индексподдержать его.

CREATE INDEX months_number_of_year_number_of_month_first_day_last_day
             ON months (number_of_year,
                        number_of_month,
                        first_day,
                        last_day);
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...