5.3. Выполнение вычислений с датами и временем

5.3. Выполнение вычислений с датами и временем

Проблема

Требуется узнать количество времени, прошедшего между двумя точками даты/времени.

Решение

Если обе временные точки находятся между 1970 и 2038 годами, то используйте тип time_t и функцию difftime, определенную в заголовочном файле <ctime>. Пример 5.6 показывает, как вычислить число дней, прошедших между двумя датами.

Пример 5.6. Вычисление даты и времени в формате time_t

#include <ctime>

#include <iostream>

#include <cstdlib>

using namespace std;

time_t dateToTimeT(int month, int day, int year) {

 // 5 января 2000 года передается как (1, 5, 2000)

 tm tmp = tm();

 tmp.tm_mday = day;

 tmp.tm_mon = month - 1;

 tmp.tm_year = year - 1900;

 return mktime(&tmp);

}

time_t badTime() {

 return time_t(-1);

}

time_t now() {

 return time(0);

}

int main() {

 time_t date1 = dateToTimeT(1,1,2000);

 time_t date2 = dateToTimeT(1,1,2001);

 if ((date1 == badTime()) || (date2 == badTime())) {

  cerr << "невозможно создать структуру time_t" << endl;

  return EXIT_FAILURE;

 }

 double sec = difftime(date2, date1);

 long days = static_cast<long>(sec / (60 * 60 — 24));

 cout << число дней между 1 января 2000 г. и 1 января 2001 г. составляет ";

 cout << days << endl;

 return EXIT_SUCCESS;

}

Программа из примера 5.6 должна вывести:

число дней между 1 января 2000 г. и 1 января 2001 г. составляет 366

Обратите внимание, что 2000 год високосный, так как, несмотря на то что он делится на 100, он также делится и на 400 и, следовательно, состоит из 366 дней.

Обсуждение

Тип time_t — это зависящий от реализации арифметический тип. Это означает, что это либо целый тип, либо тип с плавающей точкой, и, таким образом, он поддерживает основные арифметические операции. Его можно складывать, вычитать, делить, умножать и т.д. Чтобы вычислить интервал между двумя значениями time_t в секундах, используйте функцию difftime. Не думайте, что сам time_t содержит секунды, даже если это и так. Многие реализации C++ могут в ближайшем будущем молча изменить его так, чтобы он содержал доли секунд (это одна из причин, по которым difftime возвращает double).

Если ограничения time_t слишком жестки, то вместо него для вычисления временных интервалов потребуется использовать различные классы из библиотеки Boost date_time. Пример 5.7 показывает, как использовать классы Boost для вычисления числа дней в 20-м и 21-м столетиях.

Пример 5.7. Вычисление даты и времени с помощью date_duration

#include <iostream>

#include <boost/date_time/gregorian/gregorian.hpp>

using namespace std;

using namespace boost::gregorian;

int main() {

 date_duration dd = date(2000, 1, 1) - date(1900, 1, 1);

 cout << "Двадцатый век содержал " << dd.days() << " дней" << endl;

 dd = date(2100, 1, 1) - date(2000, 1, 1);

 cout << "Двадцать первый век будет содержать " <<

  dd.days() << " дней" << endl;

}

Программа из примера 5.7 должна вывести:

Двадцатый век содержал 36 524 дней

Двадцать первый век будет содержать 36 525 дней