MySQL Query GROUP BY день / месяц / год - PullRequest
579 голосов
/ 03 февраля 2009

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

SELECT COUNT(id)
FROM stats
WHERE record_date.YEAR = 2009
GROUP BY record_date.YEAR

Или даже:

SELECT COUNT(id)
FROM stats
GROUP BY record_date.YEAR, record_date.MONTH

Иметь ежемесячную статистику.

Спасибо!

Ответы [ 13 ]

916 голосов
/ 03 февраля 2009
GROUP BY YEAR(record_date), MONTH(record_date)

Проверьте функции даты и времени в MySQL.

204 голосов
/ 23 апреля 2011
GROUP BY <a href="http://dev.mysql.com/doc/refman/4.1/en/date-and-time-functions.html#function_date-format">DATE_FORMAT</a>(record_date, '%Y%m')

Примечание (в основном, для потенциальных потребителей). В настоящее время это может быть не так эффективно, как другие предложения. Тем не менее, я оставляю это как альтернативу, и еще одну, которая может помочь увидеть, насколько быстрее другие решения. (Поскольку вы не можете по-настоящему отличить быстрое от медленного до тех пор, пока не увидите разницу.) Кроме того, со временем могут быть внесены изменения в движок MySQL в отношении оптимизации, чтобы сделать это решение в некоторых (возможно, не очень отдаленная) точка в будущем, чтобы стать вполне сопоставимой по эффективности с большинством других.

40 голосов
/ 23 апреля 2011

Я пытался использовать приведенное выше утверждение «ГДЕ», я считал его правильным, поскольку никто не исправил его, но я ошибался; после некоторых поисков я обнаружил, что это правильная формула для оператора WHERE, поэтому код выглядит так:

SELECT COUNT(id)  
FROM stats  
WHERE YEAR(record_date) = 2009  
GROUP BY MONTH(record_date)
38 голосов
/ 11 октября 2011

попробуйте это

SELECT COUNT(id)
FROM stats
GROUP BY EXTRACT(YEAR_MONTH FROM record_date)

EXTRACT (единица измерения от даты) функция лучше, поскольку используется меньшее количество группировок и функция возвращает числовое значение.

Условие сравнения при группировке будет быстрее, чем функция DATE_FORMAT (которая возвращает строковое значение). Попробуйте использовать функцию | поле, которое возвращает нестроковое значение для условия сравнения SQL (WHERE, HAVING, ORDER BY, GROUP BY).

25 голосов
/ 09 июня 2013

Если ваш поиск превышает несколько лет, и вы все еще хотите группировать ежемесячно, я предлагаю:

версия № 1:

SELECT SQL_NO_CACHE YEAR(record_date), MONTH(record_date), COUNT(*)
FROM stats
GROUP BY DATE_FORMAT(record_date, '%Y%m')

версия # 2 (более эффективная) :

SELECT SQL_NO_CACHE YEAR(record_date), MONTH(record_date), COUNT(*)
FROM stats
GROUP BY YEAR(record_date)*100 + MONTH(record_date)

Я сравнил эти версии в большой таблице с 1 357 918 строками (), и вторая версия, кажется, имеет лучшие результаты.

версия1 (в среднем 10 исполнений) : 1,404 секунды
version2 (в среднем 10 выполнений) : 0,780 секунды

(добавлен ключ SQL_NO_CACHE для предотвращения кэширования MySQL запросов.)

16 голосов
/ 10 октября 2011

Если вы хотите сгруппировать по дате в MySQL, используйте следующий код:

 SELECT COUNT(id)
 FROM stats
 GROUP BY DAYOFMONTH(record_date)

Надеюсь, это сэкономит время тем, кто собирается найти эту ветку.

11 голосов
/ 18 января 2014

Если вы хотите отфильтровать записи за определенный год (например, 2000), оптимизируйте предложение WHERE следующим образом:

SELECT MONTH(date_column), COUNT(*)
FROM date_table
WHERE date_column >= '2000-01-01' AND date_column < '2001-01-01'
GROUP BY MONTH(date_column)
-- average 0.016 sec.

Вместо:

WHERE YEAR(date_column) = 2000
-- average 0.132 sec.

Результаты были сгенерированы для таблицы, содержащей 300 тыс. Строк и индекс по столбцу даты.

Что касается предложения GROUP BY, я проверил три варианта в соответствии с вышеупомянутой таблицей; Вот результаты:

SELECT YEAR(date_column), MONTH(date_column), COUNT(*)
FROM date_table
GROUP BY YEAR(date_column), MONTH(date_column)
-- codelogic
-- average 0.250 sec.

SELECT YEAR(date_column), MONTH(date_column), COUNT(*)
FROM date_table
GROUP BY DATE_FORMAT(date_column, '%Y%m')
-- Andriy M
-- average 0.468 sec.

SELECT YEAR(date_column), MONTH(date_column), COUNT(*)
FROM date_table
GROUP BY EXTRACT(YEAR_MONTH FROM date_column)
-- fu-chi
-- average 0.203 sec.

Последний победитель.

9 голосов
/ 07 сентября 2017

Полное и простое решение с аналогичным исполнением, но более короткой и более гибкой альтернативой, в настоящее время активной:

SELECT COUNT(*) FROM stats
-- GROUP BY YEAR(record_date), MONTH(record_date), DAYOFMONTH(record_date)
GROUP BY DATE_FORMAT(record_date, '%Y-%m-%d')
7 голосов
/ 22 ноября 2013

Если вы хотите получать ежемесячную статистику с количеством строк в месяц каждого года, упорядоченного по последнему месяцу, попробуйте следующее:

SELECT count(id),
      YEAR(record_date),
      MONTH(record_date) 
FROM `table` 
GROUP BY YEAR(record_date),
        MONTH(record_date) 
ORDER BY YEAR(record_date) DESC,
        MONTH(record_date) DESC
6 голосов
/ 30 ноября 2016

Вы можете сделать это просто функцией Mysql DATE_FORMAT () в GROUP BY. Возможно, вы захотите добавить дополнительный столбец для большей ясности в некоторых случаях, например, когда записи охватывают несколько лет, а один и тот же месяц встречается в разные годы. Здесь так много вариантов, которые можно настроить. Пожалуйста, прочитайте это перед началом. Надеюсь, это будет очень полезно для вас. Вот пример запроса для вашего понимания

SELECT
    COUNT(id),
    DATE_FORMAT(record_date, '%Y-%m-%d') AS DAY,
    DATE_FORMAT(record_date, '%Y-%m') AS MONTH,
    DATE_FORMAT(record_date, '%Y') AS YEAR,

FROM
    stats
WHERE
    YEAR = 2009
GROUP BY
    DATE_FORMAT(record_date, '%Y-%m-%d ');
...