Первое значение каждого месяца в SQL

#sql #timestamp

#sql #временная метка

Вопрос:

У меня есть база данных, содержащая значения данных с отметками времени, с несколькими выборками в день, т.Е.

 05/01/11 01:00:00 - 1.23  
05/01/11 01:12:34 - 0.99  
....  
05/01/11 23:59:59 - 2.34  
05/02/11 00:11:22 - 4.56
  

и т. д

Я пытаюсь получить список первого образца каждого месяца. Мне удалось получить список каждого образца 1-го числа каждого месяца:

 SELECT
    "RecordTime", "FormattedValue"
FROM
    CDBHistoric
WHERE
    "Id" = 12345 AND EXTRACT (DAY FROM "RecordTime") = 1
  

Это дает такие данные, как

 03/01/11 00:00:01 - 1.23  
03/01/11 00:12:34 - 0.99  
....  
04/01/11 00:00:34 - 2.34  
04/01/11 00:11:22 - 4.56
  

но мне не удалось получить первое значение каждой группы.

Комментарии:

1. Используете ли вы Oracle? Вы просто пытаетесь получить первую запись за каждый день?

2. Какая СУБД? Ответ в SQL Server будет сильно отличаться от ответа для Oracle, который будет сильно отличаться от ответа для MySQL и т.д. SQL Недостаточно информации для предоставления ответа.

3. @Ken White: код является стандартным SQL-92, поэтому я думаю, что sql тега достаточно (т. Е. Почему бы не опубликовать ответ, используя стандартный SQL-92?)

Ответ №1:

Вот решение для SQL Server. Идея состоит в том, чтобы использовать коррелированный подзапрос для определения минимума RecordTime для каждого года и месяца. Должно быть легко воплотить идею в соответствии с вашим вкусом СУБД.

 WITH Data AS (
    SELECT CONVERT(DATETIME, '1-May-2011 01:00:00') AS RecordTime, 1.23 AS FormattedValue
    UNION ALL SELECT CONVERT(DATETIME, '1-May-2011 01:12:34'), 0.99
    UNION ALL SELECT CONVERT(DATETIME, '1-May-2011 23:59:59'), 2.34
    UNION ALL SELECT CONVERT(DATETIME, '2-May-2011 00:11:22'), 4.56
)
SELECT D1.*
FROM Data D1
WHERE D1.RecordTime = (SELECT MIN(RecordTime)
                       FROM Data D2
                       WHERE YEAR(D2.RecordTime) = YEAR(D1.RecordTime)
                       AND MONTH(D2.RecordTime) = MONTH(D1.RecordTime))
  

Редактировать: Если ваш набор данных большой, то объединение в подзапросе должно быть существенно быстрее, но может оказаться сложнее перевести в вашу СУБД:

 SELECT D1.*
FROM Data D1
INNER JOIN (
    SELECT MIN(D2.RecordTime) AS FirstMonthlyRecordTime
    FROM Data D2
    GROUP BY YEAR(D2.RecordTime), MONTH(D2.RecordTime)
) AS SubQuery
    ON D1.RecordTime = SubQuery.FirstMonthlyRecordTime
  

Редактировать 2: Следующий код является стандартной версией SQL-92 моего первого запроса выше (протестирован на MySQL):

 SELECT T1.RecordTime, T1.FormattedValue
FROM CDBHistoric T1
WHERE T1.RecordTime = (SELECT MIN(T2.RecordTime)
                       FROM CDBHistoric T2
                       WHERE EXTRACT(YEAR FROM T1.RecordTime) = EXTRACT(YEAR FROM T2.RecordTime)
                       AND EXTRACT(MONTH FROM T1.RecordTime) = EXTRACT(MONTH FROM T2.RecordTime))
  

Комментарии:

1. Код операционной системы является стандартным SQL-92, который SQL Server не поддерживает, но AFAIK PostgreSQL, Oracle и MySQL поддерживают.

2. Я добавил версию моего первого запроса на SQL-92.