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 дней