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

C++ 如何从字符串解析日期/时间?,c++,datetime,boost,utc,boost-date-time,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偏移量也很好 假设输入是相关

输入:带有日期和可选时间的字符串。不同的表示形式很好,但却是必要的。字符串由用户提供,格式可能不正确。示例:

    >代码>“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\u 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\u time
可以提供所需的UTC偏移量,但我不知道如何提供。

最简单的便携式解决方案是使用
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 ...
使用
int
值初始化
struct tm
,并将其传递给
mktime
,以获取作为
time
的日历时间。有关时区转换,请打开
gmtime

有一些您需要的东西,无需再做任何工作:

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::gregorian;
{
//以下日期为ISO 8601扩展格式(CCYY-MM-DD)
标准::字符串s(“2000-01-01”);
日期d(来自简单字符串);
std::cout如果可以接受c-style:()是一种方法,因为您可以指定格式,并且可以考虑语言环境:

tm brokenTime;
strptime(str.c_str(), "%Y-%m-%d %T", &brokenTime);
time_t sinceEpoch = timegm(brokenTime);
必须使用返回值检查不同的布局(如果可能)。
必须通过检查系统时钟来添加时区(localtime\r()和time(),tm\u zone)

虽然我不知道如何在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");
}


您可能可以使用
boost::posix_time::c_time::localtime()
from
#include
在假设输入在当前时区的情况下完成此转换,但这是相当不一致的:例如,对我来说,夏令时结束后,今天和下个月的结果会有所不同。

我相信你必须自己解析它们(可能是使用spirit)因为“2004-3-21”中的一位数月数任何boost time IO格式说明符都无法解析@Cubbi:如果这是唯一的问题,那么检查并在字符串中插入0要比将spirit放入图片中容易得多。@Cubbi-您可以在
boost::date\u time
-
boost::spirit
中处理自定义输入和输出格式太过了here@Cubbi:如果sIGLE数字是一个问题,用户只需要输入两位数的值。我将编辑我的帖子,因为这不是真正的问题。C运行时<代码> SCANF/PrimTf<代码>介绍了使用适当的C++库可以避免的缓存管理和类型安全问题。这不能解决本地到UTC的问题。ING是用户提供的,并且可能是无效的/畸形的。我认为这可能是SCANF的问题。@加布里埃尔,如果字符串是畸形的,SCANF将不会返回6。关于UTC,我已经向答案添加了更多的信息。@史提夫,请指出在这种特殊情况下可能出现的缓冲管理/类型安全问题和它们的C++ ALTE。活动?这将是有益的。(x)如果格式字符串与计数或类型的输入不匹配,那么任何人都会表现得不可预知。为什么在C++中有人使用Prtff/ScNf?如果Q是标记C,则这是不存在的。StrpTimes已经被尝试了。它是不可接受的,因为如果字符串不好,它会很高兴地崩溃。我用它,它不会在这里崩溃,但是经验可能会不同。我必须(向谷歌)调查以确保…
strtime
也是Linux专用的,据我所知。呃,让POSIX专用。@KyleStrand不用担心,顺便提一下,问题提到“仅限Linux”.Thx用于facet/tutorial linx。使用boost::gregorian无法解决此问题,因为它不提供时间解析/表示。@Gabriel-是的,您必须使用这些工具构建自己的解析器和格式化程序来处理所有需要的情况。除非您有无限可能的输入格式,否则使用pa应该可以实现这一点每种格式的rser,以及一个标识格式类型并提供给相应解析器的包装器。@Gabriel-注意,当我说parser时,考虑到您的输入
字符串
选项,这其实并不复杂。只需检测每种格式并构建相应的Boost结构,以适当地解析为日期时间。演示如何使用fa非常感谢cets。如果我理解正确,使用localtime不是一个选项,因为它会给我今天的DST偏移量,而不是给定日期的DST偏移量。@Gabriel Schreiber:您可能可以在给定日期进行DST偏移量,方法与
utc_to_local()相反
/usr/include/boost/date\u time/c\u local\u time\u adjustor.hpp中执行,它仍将使用当前计算机的区域。更好的方法可能是更接近
~ $ ./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