Разбор временной метки RFC3339 / ISO 8601 в режиме Boost

Как мне разобрать RFC3339 метка времени («1985-04-12T23: 20: 50.52Z») (то есть подмножество ISO8601) в C ++ 03? Я использую Boost, но ни одна из библиотек даты и времени Boost, кажется, не включает функцию для этого.

Тип объекта реального времени не имеет значения, если я могу легко сравнить его с «сейчас». Меня волнуют только временные метки в часовом поясе UTC.

1

Решение

Без использования Boost, просто strptime вы можете.

Предполагая, что тот же самый _adaptive_parser_ помощник размещен здесь: Использование строки разбора даты и времени: с однозначным часовым форматом

Примечание: образцы взяты из ссылки RFC

#include "adaptive_parser.h"#include <string>
#include <iostream>

int main() {
using namespace mylib::datetime;

adaptive_parser parser;

for (std::string const input : {
"1985-04-12T23:20:50.52Z",
"1996-12-19T16:39:57-08:00",
"1990-12-31T23:59:60Z",
"1990-12-31T15:59:60-08:00",
"1937-01-01T12:00:27.87+00:20",
})
try {
std::cout << "Parsing '" << input << "'\n";
std::cout << " -> epoch " << parser(input).count() << "\n";
} catch(std::exception const& e) {
std::cout << "Exception: " << e.what() << "\n";
}
}

Печать

Parsing '1985-04-12T23:20:50.52Z'
-> epoch 482196050
Parsing '1996-12-19T16:39:57-08:00'
-> epoch 851042397
Parsing '1990-12-31T23:59:60Z'
-> epoch 662688000
Parsing '1990-12-31T15:59:60-08:00'
-> epoch 662688000
Parsing '1937-01-01T12:00:27.87+00:20'
-> epoch -1041335973

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

1

Другие решения

Есть ограничение разбора часового пояса.

#include <sstream>
#include <iostream>
#include <string>
#include <iomanip>

int main() {
std::tm t = {};
std::string s = "2016-01-02T15:04:05+09:00";
int tz_offset = 0;
auto pos = s.find_last_of("+-Z");
if (pos != s.npos) {
std::string zone = s.substr(pos);
tz_offset = std::atoi(zone.c_str());
s.resize(pos);
}
std::stringstream ss(s);
ss >> std::get_time(&t, "%Y-%m-%dT%H:%M:%S");
if (ss.fail()) {
std::cout << "Parse failed\n";
} else {
std::time_t l = std::mktime(&t);
std::tm tm_utc(*std::gmtime(&l));
std::time_t utc = std::mktime(&tm_utc);
tz_offset += (utc - l);
l = std::mktime(&t) - tz_offset;
t = *std::gmtime(&l);
std::cout << std::put_time(&t, "%c") << std::endl;
}
}
0

По вопросам рекламы [email protected]