如何从字符串中解析日期/时间?

时间:2010-09-24 10:37:26

标签: c++ datetime boost utc boost-date-time

输入:包含日期和可选时间的字符串。不同的表现形式会很好但很必要。字符串是用户提供的,可能会格式错误。例子:

  • "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 个答案:

答案 0 :(得分:65)

虽然我不知道如何在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>来完成此转换,假设输入位于当前时区,但它相当不一致:例如,对我来说,结果会有所不同在今天和下个月之间,当夏令时结束时。

答案 1 :(得分:11)

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;
}

有关如何使用boost::posix_time here的UTC偏移量的示例。

您可以使用date_input_facettime_input_facet从自定义输入字符串格式生成日期和时间。 <{3}}上有一个I / O教程可以帮助您开始工作。

答案 2 :(得分:8)

如果可以接受c风格:strptime()是可行的方法,因为您可以指定格式并且可以在帐户中使用区域设置:

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

必须使用返回值检查不同的布局(如果可能)。 必须通过检查系统时钟(localtime_r()with time(),tm_zone)来添加时区

答案 3 :(得分:3)

最简单,可移植的解决方案是使用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上的see information