6  Работа со временем

library(tidyverse)

Переменные бывают разные, и о них можно думать как о шкале:

Кажется, что время — просто обычная числовая переменная, на которой определены все обычные операции сложения вычитания и т. п. Однако стоит держать в голове несколько фактов:

Данные расхождения с ожиданиями связаны с двумя возможными определениями суток:

Так как секунду в какой-то момент определили без привязки к вращению Земли, ученым приходится периодически подкручивать время, добавляя високосные секунды.

Все это мелочи учтены в пакете lubridate, созданном для работы со временем в R (https://rawgit.com/rstudio/cheatsheets/master/lubridate.pdf, туториал доступен здесь и по команде vignette("lubridate")). Первые функции, которые нужно изучить, это today() и now():

library(lubridate)
today()
[1] "2025-01-16"
now()
[1] "2025-01-16 05:35:33 MSK"

Как видно из этих функций, в R можно работать как с датами, так и со временем. В качестве иллюстрации мы будем использовать датасет flights из пакета nycflights13, в котором содержатся данные полетов из Нью Йорка в 2013 году.

library(nycflights13)
flights

6.1 Создание даты

Самый простой способ получить дату — это преобразовать строку в формат даты. Для этого надо просто упорядочить y (year), m (month) и d (day) в команде:

ymd("2020-01-21")
[1] "2020-01-21"
ymd("20-01-21")
[1] "2020-01-21"
ymd("20.01.21")
[1] "2020-01-21"
ymd("20/01/21")
[1] "2020-01-21"
ymd("200121")
[1] "2020-01-21"
mdy("January 21st, 2020")
[1] "2020-01-21"
dmy("21-Jan-2020")
[1] "2020-01-21"

Команды понимают не только английский (хоть и с трудом):

dmy("21 янв 2020", locale = "ru_RU.UTF-8")
[1] "2020-01-21"
dmy("21 янв. 2020", locale = "ru_RU.UTF-8")
[1] "2020-01-21"
dmy("21 ян 2020", locale = "ru_RU.UTF-8")
Warning: All formats failed to parse. No formats found.
[1] NA
dmy("21 янва 2020", locale = "ru_RU.UTF-8")
[1] "2020-01-21"
dmy("21 января 2020", locale = "ru_RU.UTF-8")
Warning: 1 failed to parse.
[1] NA
dmy("21 январь 2020", locale = "ru_RU.UTF-8")
[1] "2020-01-21"
dmy("21 Январь 2020", locale = "ru_RU.UTF-8")
[1] "2020-01-21"

Аналогично сделаны команды состоящие из h, m, s:

hms("20:01:02")
[1] "20H 1M 2S"
hm("20.01")
[1] "20H 1M 0S"
ms("23:59")
[1] "23M 59S"

Также существует команда make_datetime(), которая позволяет сделать дату из нескольких переменных:

flights |> 
  mutate(departure = make_datetime(year, month, day, hour, minute)) |> 
  select(departure)

6.2 Работа с часовыми поясами

Земля разбита на условные географическо-административные зоны, в которых действуют свои правила работы со временем. В каких-то зонах есть переход на зимнее/летнее время, а где-то его нет. В некоторых точках Земли понятие часового пояса не имеет смысла, однако все равно есть конвенции того, какое время на этой территории использовать. Функция make_datetime(), которую мы рассмотрели, использует по-умолчанию всемирное координированное время (UTC). Обозначение интересующего часового пояса можно посмотреть в интернете, однако основная информации о возможных значениях аргумента tz хранится в системе пользователя.

dmy_hm("21-01-2001 15^43", tz = "Europe/Moscow")
[1] "2001-01-21 15:43:00 MSK"
dmy_hm("21-01-2001 15^43", tz = "America/Chicago")
[1] "2001-01-21 15:43:00 CST"
dmy_hm("21-01-2001 15^43", tz = "America/New_York")
[1] "2001-01-21 15:43:00 EST"
dmy_hm("21-01-2001 15^43", tz = "Africa/Cairo")
[1] "2001-01-21 15:43:00 EET"

6.3 Извлечение компонентов даты

Для извлечения компонентов даты используются функции year(), month(), week() (номер недели в году), mday() (day of the month), wday() (номер дня в неделе), yday() (номер дня в году), hour(), minute() и second():

date_example <- flights$time_hour[1]
date_example
[1] "2013-01-01 05:00:00 EST"
year(date_example)
[1] 2013
month(date_example)
[1] 1
month(date_example, label = TRUE)
[1] Jan
12 Levels: Jan < Feb < Mar < Apr < May < Jun < Jul < Aug < Sep < ... < Dec
month(date_example, label = TRUE, abbr = FALSE)
[1] January
12 Levels: January < February < March < April < May < June < ... < December
month(date_example, label = TRUE, locale = "ru_RU.UTF-8")
[1] янв
12 Levels: янв < фев < мар < апр < мая < июн < июл < авг < сен < ... < дек
week(date_example)
[1] 1
mday(date_example)
[1] 1
wday(date_example)
[1] 3
wday(date_example, label = TRUE)
[1] Tue
Levels: Sun < Mon < Tue < Wed < Thu < Fri < Sat
wday(date_example, label = TRUE, abbr = FALSE)
[1] Tuesday
7 Levels: Sunday < Monday < Tuesday < Wednesday < Thursday < ... < Saturday
wday(date_example, label = TRUE, locale = "ru_RU.UTF-8")
[1] Вт
Levels: Вс < Пн < Вт < Ср < Чт < Пт < Сб
yday(date_example)
[1] 1
hour(date_example)
[1] 5
minute(date_example)
[1] 0
second(date_example)
[1] 0

Также есть функция leap_year(), которая сообщает, является ли выбранный год високосным:

leap_year(2019)
[1] FALSE
leap_year(2020)
[1] TRUE

6.4 Операции с датами

Если взять две даты, то можно узнать разницу между ними и т. п.:

ymd("2020-01-21") - ymd("2020-01-19")
Time difference of 2 days
ymd("2020-01-19") - ymd("2020-01-21")
Time difference of -2 days

Обратите внимание на результат работы этого выражения:

hm("21:00") - hm("18:10")
[1] "3H -10M 0S"

Видимо, почему-то при таком использовании происходит поэлементная операция с часами, минутами, и секундами, так что в результате получаются отрицательные минуты. Однако, если использовать полные даты, то этого эффекта нет:

ymd_hm("2020-01-21, 21:00") - ymd_hm("2020-01-21, 18:10")
Time difference of 2.833333 hours
ymd_hm("2020-01-21, 21:00") - hm("18:10")
[1] "2020-01-21 02:50:00 UTC"

Также существует функция difftime(), которая позволяет настраивать единицы, в которых выдается результат:

difftime(ymd_hm("2020-01-21, 21:00"), ymd_hm("2020-01-21, 18:10"), units = "mins")
Time difference of 170 mins
difftime(ymd_hm("2020-01-21, 21:00"), ymd_hm("2020-01-21, 18:10"), units = "hours")
Time difference of 2.833333 hours

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

  • periods — промежутки времени, которые игнорируют нерегулярности во времени, сразу прибавляя 1 к соответствующему разряду, вводятся функциями years(), months(), weeks(), days(), hours(), minutes(), seconds(), period()
  • duration — промежутки времени, которые учитывают нерегулярности во времени, добавляя стандартную длительность единицы, вводятся функциями dyears(), dweeks(), ddays(), dhours(), dminutes(), dseconds(), duration()

Рассмотрим несколько сложных случаев:

  • високосный год
ymd("2019-03-01")+years(1)
[1] "2020-03-01"
ymd("2019-03-01")+dyears(1)
[1] "2020-02-29 06:00:00 UTC"
  • переход на летнее время
ymd_hms("2020-03-07 13:00:00", tz = "America/New_York") + days(1)
[1] "2020-03-08 13:00:00 EDT"
ymd_hms("2020-03-07 13:00:00", tz = "America/New_York") + ddays(1)
[1] "2020-03-08 14:00:00 EDT"
  • переход на зимнее время
ymd_hms("2020-10-31 13:00:00", tz = "America/New_York") + days(1)
[1] "2020-11-01 13:00:00 EST"
ymd_hms("2020-10-31 13:00:00", tz = "America/New_York") + ddays(1)
[1] "2020-11-01 12:00:00 EST"

Последняя операция с датами, которую мы рассмотрим — округление:

  • floor_date() — округление в меньшую сторону
  • round_date() — математическое округление
  • ceiling_date() — округление в большую сторону

floor_date(ymd("2020-01-16"), unit = "month")
[1] "2020-01-01"
round_date(ymd("2020-01-16"), unit = "month")
[1] "2020-01-01"
round_date(ymd("2020-01-17"), unit = "month")
[1] "2020-02-01"
ceiling_date(ymd("2020-01-16"), unit = "month")
[1] "2020-02-01"
ceiling_date(ymd("2020-01-16"), unit = "year")
[1] "2021-01-01"

6.5 Визуализация времени: данные Левада-центра

Пакет tidyverse понимает переменные типа дата, и позволяет их фильтровать и визуализировать. Возьмем для примера датасет из проекта The Unwelcomed Мохамада А. Вэйкда (Mohamad A. Waked), содержащий информацию о месте и причинах смерти мигрантов и беженцев по всему миру с января 2014 года по июнь 2019 года.

unwelcomed <- read_csv("https://raw.githubusercontent.com/agricolamz/daR4hs/main/data/w6_death_of_migrants_and_refugees_from_the_Unwelcomed_project.csv")
unwelcomed |> 
  mutate(date = dmy(date)) |> 
  ggplot(aes(date, total_death_missing, color = collapsed_cause))+
  geom_point()+
  scale_y_log10()+
  labs(y = "number of death/missing")
Warning in scale_y_log10(): log-10 transformation introduced infinite values.

unwelcomed |> 
  mutate(date = dmy(date)) |> 
  filter(date < dmy("1-1-2016")) |> 
  ggplot(aes(date, total_death_missing, color = collapsed_cause))+
  geom_point()+
  scale_y_log10()+
  labs(y = "number of death/missing")
Warning in scale_y_log10(): log-10 transformation introduced infinite values.

Однако к переменным со временем не всегда относятся аккуратно. Рассмотрим график с сайта Левада-центра — российской негосударственной исследовательской организации, которая проводит социологические и маркетинговые исследования (график взят отсюда):

На первый взгляд, в этом графике нет ничего странного, однако если присмотреться к динамической версии на сайте Левада-центра, можно обнаружить, что не идущие подряд значения расположены на одинаковом расстоянии друг от друга, например, 05.2014, 07.2014, 11.2014. Вот здесь можно скачать данные, по которым строился этот график. Вот как он выглядит, если считать временную переменную как время

levada <- read_csv("https://raw.githubusercontent.com/agricolamz/daR4hs/main/data/w6_2019.01_levada_countries.csv")

levada |> 
  mutate(date = str_c("1-", date),
         date = dmy(date)) |> 
  filter(towards == "USA") |> 
  pivot_longer(names_to = "answer", values_to = "number", good:bad) |> 
  ggplot(aes(date, number, color = answer))+
  geom_line()+
  labs(x = "", y = "", caption = "данные Левада-центра")+
  scale_y_continuous(limits = c(0, 100))+
  theme(legend.position = c(0.1, 0.9), legend.title = element_blank())

На графике теперь видно, насколько регулярно проводились опросы: в начале 90-ых опросы проводились реже, потом часто, потом в районе 2010 года был перерыв. График Левада-центра можно оправдать тем, что они представляют данные от замера к замеру, так что по оси x находится как бы категориальная переменная со значениями замер 05.2014, замер 07.2014, замер 11.2014 и т. д. Однако это совсем неочевидно из графика.