boost :: date_time (boost-145) с использованием 64-битной uint с микросекундными вычислениями, без усечения - PullRequest
5 голосов
/ 15 декабря 2010

Я использую date_time, чтобы абстрагироваться от особенностей платформы.и мне нужно создать 64-битное микросекундное разрешение uint64_t, которое будет использоваться при сериализации.Я не понимаю, что происходит ниже.

#include <boost/date_time/posix_time/posix_time.hpp>
#include <boost/cstdint.hpp>
#include <iostream>

using namespace boost::posix_time;
using boost::uint64_t;

ptime UNIX_EPOCH(boost::gregorian::date(1970,1,1));

int main() {
    ptime current_time = microsec_clock::universal_time();

    std::cout << "original time: "<< current_time << std::endl;

    long microsec_since_epoch = ((current_time -UNIX_EPOCH).total_microseconds());

    ptime output_ptime = UNIX_EPOCH + microseconds(microsec_since_epoch);
        std::cout << "Deserialized time : " << output_ptime << std::endl;

    std::cout << "Microsecond output: " << microsec_since_epoch << std::endl;

    std::cout << "Microsecond to second arithmetic: "
        << microsec_since_epoch/(10*10*10*10*10*10) << std::endl;

    std::cout << "Microsecond to tiume_duration, back to microsecond : " <<
        microseconds(microsec_since_epoch).total_microseconds() << std::endl;


    return 0;
}

Вот вывод, который я получаю.

original time: 2010-Dec-17 09:52:06.737123
Deserialized time : 1970-Jan-16 03:10:41.577454
Microsecond output: 1292579526737123
Microsecond to second arithmetic: 1292579526
Microsecond to tiume_duration, back to microsecond : 1307441577454

Когда я переключаюсь на использование total_seconds() и + seconds(..) Проблемы исчезают --ie, ввод изменяется на:

2010-Dec-15 18:26:22.606978
2010-Dec-15 18:26:22

date_time утверждает, что использует 64-битвведите внутренне, и 2^64÷ (10^6×3600×24×365) ~= 584942 даже 2^60÷ (10^6×3600×24×365) ~= 36558.

В открывающих строках из Википедии есть это, чтобы сказать о времени Posix

Время Unix, или время POSIX, является системойописывающие моменты времени, определяемые как количество секунд, прошедших с полуночи всемирного координированного времени (UTC) на 1 января 1970 года

Почему такое массовое усечение происходит 40 лет спустя?

Как использовать полное 64-битное пространство с микросекундным разрешением с использованием boost :: date_time?

- edit1 в ответ на hans -

Запись была изменена для отображения целочисленного вывода части duration.total_microseconds ().Примечание 1292576572566904 ÷ (10 ^ 6 × 3600 × 24 × 365) ~ = 40,98 года.Выходные данные в секундах не обновлялись.

- edit2 - Уменьшение микросекунд до секунд перед этапом «десериализации» также работает хорошо.Этот подход решил мою проблему, мне нужно только микросекундное разрешение при создании, и я могу жить без него при десериализации.

Я все еще хочу знать, что и почему проблемы.

Ответы [ 2 ]

2 голосов
/ 17 декабря 2010

Кажется, это проблема того, что микросекунды () не могут обработать такой большой ввод микросекунд. Следующий фрагмент исправляет эту проблему:

#define MICROSEC 1000000

uint64_t sec_epoch = microsec_since_epoch / MICROSEC;
uint64_t mod_micro_epoch= microsec_since_epoch % MICROSEC;

ptime new_method = UNIX_EPOCH  + seconds(sec_epoch) + microseconds(mod_micro_epoch);

std::cout << "Deserialization with new method: " << new_method << std::endl;
1 голос
/ 15 декабря 2010

Тип возвращаемого значения для total_microseconds (): tick_type, не длинный. Похоже, вы компилируете это с помощью компилятора, который имеет 32-битный тип long. Очень много, чтобы хранить 40 лет микросекунд.

...