Как разобрать дату / время из строки? - PullRequest
52 голосов
/ 24 сентября 2010

Ввод : строки с датой и дополнительным временем.Разные представления были бы хорошими, но необходимыми.Строки предоставляются пользователем и могут быть повреждены.Примеры:

  • "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, но я не знаю, как.

Ответы [ 4 ]

65 голосов
/ 24 сентября 2010

Хотя я не знаю, как отформатировать однозначный ввод месяца в 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>, чтобы выполнить это преобразование, предполагая, что входные данные находятся в текущем часовом поясе, но это довольно противоречиво:для меня, например, результат будет отличаться между сегодняшним и следующим месяцем, когда заканчивается летнее время.

11 голосов
/ 24 сентября 2010

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. на этой странице есть руководство по вводу / выводу, которое должно помочь вам начать работу.

7 голосов
/ 24 сентября 2010

Если приемлем c-стиль: strptime () - это путь, потому что вы можете указать формат и принять во внимание локаль:

tm brokenTime;
strptime(str.c_str(), "%Y-%m-%d %T", &brokenTime);
time_t sinceEpoch = timegm(brokenTime);

Различные макетынужно будет проверить с возвращаемым значением (если это возможно).Необходимо добавить часовой пояс, проверив системные часы (localtime_r () с помощью time (), tm_zone)

2 голосов
/ 24 сентября 2010

самое простое, портативное решение - использовать 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 ...

Инициализируйте struct tm значениями int и передайте его mktime, чтобы получить календарное время как time_t. Для перевода часового пояса, пожалуйста, см. Информацию на gmtime.

...