MYSQL Дата Время Округ до Ближайшего часа - PullRequest
37 голосов
/ 13 марта 2012

У меня есть поле даты и времени в базе данных MySQL, и я хочу вывести результат с точностью до часа.

Например, 2012-04-01 00:00:01 должно читать 2012-04-01 00:00:00

Ответы [ 8 ]

44 голосов
/ 13 марта 2012

Обновление: я думаю, https://stackoverflow.com/a/21330407/480943 - лучший ответ.


Вы можете сделать это с некоторой арифметикой даты:

SELECT some_columns,
    DATE_ADD(
        DATE_FORMAT(the_date, "%Y-%m-%d %H:00:00"),
        INTERVAL IF(MINUTE(the_date) < 30, 0, 1) HOUR
    ) AS the_rounded_date
FROM your_table

Пояснения:

  • DATE_FORMAT : DATE_FORMAT(the_date, "%Y-%m-%d %H:00:00") возвращает усеченную дату до ближайшего часа (устанавливает минуты и секунды на ноль).

  • МИНУТА : MINUTE(the_date) получает значение даты в минутах.

  • IF : Это условно; если значение в параметре 1 равно true, то он возвращает параметр 2, в противном случае он возвращает параметр 3. Поэтому IF(MINUTE(the_date) < 30, 0, 1) означает «Если минутное значение меньше 30, верните 0, в противном случае верните 1». Это то, что мы будем использовать для округления - это количество часов, которое нужно добавить.

  • DATE_ADD : добавляет количество часов для раунда в результат.

29 голосов
/ 24 января 2014

Пол часа это 30 минут.Просто добавьте 30 минут к отметке времени и обрежьте минуты и секунды.

SELECT DATE_FORMAT(DATE_ADD(timestamp_column, INTERVAL 30 MINUTE),'%Y-%m-%d %H:00:00') FROM table

12 голосов
/ 02 октября 2012

первое решение soul усекается вместо округления, а второе решение не работает с такими случаями перехода на летнее время, как:

select FROM_UNIXTIME(UNIX_TIMESTAMP('2012-03-11 2:14:00') - MOD(UNIX_TIMESTAMP('2012-03-11 2:14:00'),300));

Вот альтернативный метод (1):

DATE_ADD(
    tick,
    INTERVAL (IF((MINUTE(tick)*60)+SECOND(tick) < 1800, 0, 3600) - (MINUTE(tick)*60)+SECOND(tick)) SECOND
)

Если вам не нужно беспокоиться о секундах, вы можете упростить это следующим образом (2):

DATE_ADD(
    tick,
    INTERVAL (IF(MINUTE(tick) < 30, 0, 60) - MINUTE(tick)) MINUTE
)

Или, если вы предпочитаете усекать вместо раунда, вот более простая версия метода души (3):

DATE_SUB(tick, INTERVAL MINUTE(tick)*60+SECOND(tick) SECOND)

РЕДАКТИРОВАТЬ: Я профилировал некоторые из этих запросов на своем локальном компьютере и обнаружил, что для 100 000 строк среднее время было следующим:

  • Soul's UNIXTIME метод: 0,0423 мс (быстро, но не работает с DST)
  • Мой метод 3: 0,1255 мс
  • Мой метод 2: 0,1289 мс
  • Метод Бена Ли DATE_FORMAT: 0,1495 мс
  • Мой метод 1: 0,1506 мс
6 голосов
/ 13 марта 2012

С Как округлить DateTime в MySQL? :

Это немного неприятно, когда вы делаете это с типами данных datetime;хороший кандидат для сохраненной функции.

DATE_SUB(DATE_SUB(time, INTERVAL MOD(MINUTE(time),5) MINUTE ), 
         INTERVAL SECOND(time) SECOND)

Проще, когда вы используете временные метки UNIXTIME, но это ограничено диапазоном дат 1970 - 2038.

FROM_UNIXTIME(UNIX_TIMESTAMP(time) - MOD(UNIX_TIMESTAMP(time),300))

Удачи.

1 голос
/ 23 января 2019

Чтобы округлить до текущего часа, выберите:

FROM_UNIXTIME(FLOOR(UNIX_TIMESTAMP(column_name) / 3600) * 3600).

Значение выражено в текущем часовом поясе doc

1 голос
/ 25 марта 2013

Это вернет следующий час, то есть '2012-01-02 18:02:30' будет преобразовано в '2012-01-02 19:00:00'

TIMESTAMPADD(HOUR,
    TIMESTAMPDIFF(HOUR,CURDATE(),timestamp_column_name),
    CURDATE())

Вместо CURDATE() вы можете использовать произвольную дату, например '2000-01-01' Не уверенесли могут возникнуть проблемы с использованием CURDATE(), если системная дата изменяется между двумя вызовами функции, не знаю, будет ли Mysql вызывать оба одновременно.

, чтобы получить ближайший час будет:

TIMESTAMPADD(MINUTE,
    ROUND(TIMESTAMPDIFF(MINUTE,CURDATE(),timestamp_column_name)/60)*60,
    CURDATE())

при изменении 60 на 15 будет получен ближайший 15-минутный интервал, с помощью ВТОРОГО вы можете получить ближайший желаемый второй интервал и т. Д.

Чтобы получить предыдущийчас использования TRUNCATE() или FLOOR() вместо ROUND().

Надеюсь, это поможет.

0 голосов
/ 12 марта 2018

Я думаю, что это лучший способ, так как он также будет использовать наименьшее количество ресурсов-

date_add(date(date_completed), interval hour(date_completed) hour) as date_hr
0 голосов
/ 30 ноября 2017

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

SELECT TIME_FORMAT(
  ADDTIME(
    TIMEDIFF('16:15', '10:00'), '00:59:00'
  ),
  '%H:00:00'
)
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...