提升日期解析 29FEB

boost date parse 29FEB

日期时间解析存在一些问题,我不知道有什么好的解决方法。

考虑以下代码:

#include <iostream>
#include <string>
#include <boost/date_time/gregorian/gregorian.hpp>

namespace dt = boost::gregorian;

dt::date parse_date(const std::string& msg, const std::string& format)
{
   std::stringstream s(msg);
   dt::date_input_facet * f = new dt::date_input_facet();
   if (format.empty())
   {
      f->set_iso_format();
   }
   else
   {
      f->format(format.c_str());
   }
   std::locale l(std::locale(), f);
   s.imbue(l);
   dt::date result;
   s >> result;
   return result;
}

int main()
{
    const std::string v = "0229";
    auto p = parse_date(v, "%m%d");
    std::cout << p << std::endl;
}

Live example

问题是,日期时间解析器的默认年份是 1400,不是闰年,所以今年没有 2 月 29 日。主要问题当然是为什么 1400 年是默认年份而不是任何闰年,无论如何我需要一些好的解决方法,有什么想法吗?

我认为 "why" 问题不是特别相关¹。这就是它的工作方式。关键是你知道是什么阻止你解析它。

我会在下面做简单的事情,或者 just write a tiny spirit parser (int_parser<unsigned char, 10, 2, 2> twice) 并直接构造一个 ptime

Live On Coliru

#include <iostream>
#include <string>
#include <boost/date_time/gregorian/gregorian.hpp>

namespace dt = boost::gregorian;

dt::date parse_date_mmdd(const std::string &msg) {
    std::stringstream s("2000" + msg);
    dt::date_input_facet *f = new dt::date_input_facet();
    f->format("%Y%m%d");

    std::locale l(std::locale(), f);
    s.imbue(l);
    dt::date result;
    s >> result;

    return result;
}

int main() {
    const std::string v = "0229";
    auto p = parse_date_mmdd(v);
    std::cout << p << std::endl;
}

版画

2000-Feb-29

¹ 答案可能集中在 "there is no data type associated with DayOfYear" 作为 DayOfMonth 元组(在日期处理库中没有用处)。所以你总是要根据上下文来决定年份。就像您总是必须决定在哪个时区解释 posix 时间。甚至要使用哪个日历模型。