Как анализировать дату/время из строки?
Ввод: строки с датой и необязательным временем. Различные представления были бы хорошими, но необходимыми. Строки поставляются пользователем и могут быть искажены. Примеры:
-
"2004-03-21 12:45:33"
(я считаю это макет по умолчанию)
-
"2004/03/21 12:45:33"
(необязательный макет)
-
"23.09.2004 04:12:21"
(немецкий формат, необязательный)
-
"2003-02-11"
(время может отсутствовать)
Необходимый результат: секунды с эпохи (1970/01/01 00:00:00) или другая фиксированная точка.
Бонус. Кроме того, чтение UTC-смещения локального системного времени было бы большим.
Предполагается, что ввод считается местным временем на рассматриваемой машине.
Выход должен быть UTC. Система - только Linux (нужны Debian Lenny и Ubuntu).
Я попытался использовать boost/date_time
, но должен признать, что я не могу обернуть голову вокруг документации. Следующие работы без необходимого преобразования из локального времени системы в UTC:
std::string date = "2000-01-01";
boost::posix_time::ptime ptimedate = boost::posix_time::time_from_string(date);
ptimedate += boost::posix_time::hours(Hardcoded_UTC_Offset);// where to get from?
struct tm = boost::posix_time::to_tm(ptimedate);
int64_t ticks = mktime(&mTmTime);
Я думаю, что boost::date_time
может обеспечить необходимое смещение UTC, но я не знаю, как это сделать.
Ответы
Ответ 1
Хотя я не знаю, как отформатировать однозначный месяц ввода в boost, я могу сделать это после двухзначного редактирования:
#include <iostream>
#include <boost/date_time.hpp>
namespace bt = boost::posix_time;
const std::locale formats[] = {
std::locale(std::locale::classic(),new bt::time_input_facet("%Y-%m-%d %H:%M:%S")),
std::locale(std::locale::classic(),new bt::time_input_facet("%Y/%m/%d %H:%M:%S")),
std::locale(std::locale::classic(),new bt::time_input_facet("%d.%m.%Y %H:%M:%S")),
std::locale(std::locale::classic(),new bt::time_input_facet("%Y-%m-%d"))};
const size_t formats_n = sizeof(formats)/sizeof(formats[0]);
std::time_t pt_to_time_t(const bt::ptime& pt)
{
bt::ptime timet_start(boost::gregorian::date(1970,1,1));
bt::time_duration diff = pt - timet_start;
return diff.ticks()/bt::time_duration::rep_type::ticks_per_second;
}
void seconds_from_epoch(const std::string& s)
{
bt::ptime pt;
for(size_t i=0; i<formats_n; ++i)
{
std::istringstream is(s);
is.imbue(formats[i]);
is >> pt;
if(pt != bt::ptime()) break;
}
std::cout << " ptime is " << pt << '\n';
std::cout << " seconds from epoch are " << pt_to_time_t(pt) << '\n';
}
int main()
{
seconds_from_epoch("2004-03-21 12:45:33");
seconds_from_epoch("2004/03/21 12:45:33");
seconds_from_epoch("23.09.2004 04:12:21");
seconds_from_epoch("2003-02-11");
}
обратите внимание, что вывод секунд-от-эпохи будет предполагать, что дата указана в формате UTC:
~ $ ./test | head -2
ptime is 2004-Mar-21 12:45:33
seconds from epoch are 1079873133
~ $ date -d @1079873133
Sun Mar 21 07:45:33 EST 2004
Вы могли бы использовать boost::posix_time::c_time::localtime()
из #include <boost/date_time/c_time.hpp>
, чтобы получить это преобразование, если предположить, что вход находится в текущем часовом поясе, но он довольно непоследовательный: для меня, например, результат будет отличаться между сегодняшним и следующим месяц, когда заканчивается переход на летнее время.
Ответ 2
boost::gregorian
содержит некоторые вещи, которые вам нужны, если вы не сделаете больше работы:
using namespace boost::gregorian;
{
// The following date is in ISO 8601 extended format (CCYY-MM-DD)
std::string s("2000-01-01");
date d(from_simple_string(s));
std::cout << to_simple_string(d) << std::endl;
}
Существует пример того, как использовать смещения UTC с boost::posix_time
здесь.
Вы можете генерировать дату и время из пользовательских форматов входных строк, используя date_input_facet
и time_input_facet
. На этой странице есть учебник ввода-вывода, который должен помочь вам начать работу.
Ответ 3
Если c-стиль допустим: strptime() - это путь, потому что вы можете указать формат, и он может принимать язык в аккаунте:
tm brokenTime;
strptime(str.c_str(), "%Y-%m-%d %T", &brokenTime);
time_t sinceEpoch = timegm(brokenTime);
Различные макеты должны быть проверены с возвращаемым значением (если это возможно).
Часовой пояс должен быть добавлен путем проверки системных часов (localtime_r() с помощью time(), tm_zone)
Ответ 4
Самое простое портативное решение - использовать scanf
:
int year, month, day, hour, minute, second = 0;
int r = 0;
r = scanf ("%d-%d-%d %d:%d:%d", &year, &month, &day,
&hour, &minute, &second);
if (r == 6)
{
printf ("%d-%d-%d %d:%d:%d\n", year, month, day, hour, minute,
second);
}
else
{
r = scanf ("%d/%d/%d %d:%d:%d", &year, &month, &day,
&hour, &minute, &second);
// and so on ...
Инициализировать a struct tm
с помощью значений int
и передать его на mktime
, чтобы получить календарное время как time_t
. Для преобразования часовых поясов, пожалуйста, см. Информацию в gmtime
.