c++ - 如何在 C++ 中将 iso 8601 日期(可选毫秒)解析为 struct tm?

标签 c++ parsing posix datetime-format

我有一个字符串,应该在 ISO 8601 中指定日期和时间格式,其中可能有也可能没有毫秒,我想从中获取 struct tm 以及可能已指定的任何毫秒值(如果字符串中不存在)。

检测字符串的格式是否正确,以及将用户指定的字符串转换为 struct tm 和毫秒值会涉及什么?

如果不是毫秒问题,我可能只使用 C 函数 strptime(),但我不知道该函数的定义行为应该是什么秒包含小数点。

作为最后一个警告,如果可能的话,我非常喜欢一个不依赖于仅在 Boost 中找到的函数的解决方案(但我很高兴接受 C++11 作为先决条件)。

输入看起来像:

2014-11-12T19:12:14.505Z

2014-11-12T12:12:14.505-5:00

Z,在这种情况下,表示 UTC,但可以使用任何时区,并将表示为与 GMT 的 + 或 - 小时/分钟偏移量。秒字段的小数部分是可选的,但它可能存在的事实就是为什么我不能简单地使用 strptime()std::get_time() ,如果在字符串的秒部分找到这样的字符,则不描述任何特定定义的行为。

最佳答案

旧问题的新答案。理由:更新工具。

使用这个free, open source library , 可以解析成 std::chrono::time_point<system_clock, milliseconds> ,它比 tm 更有优势能够保持毫秒精度。如果你真的需要,你可以通过 system_clock::to_time_t 继续使用 C API。 (沿途损失毫秒)。

#include "date.h"
#include <iostream>
#include <sstream>

date::sys_time<std::chrono::milliseconds>
parse8601(std::istream&& is)
{
    std::string save;
    is >> save;
    std::istringstream in{save};
    date::sys_time<std::chrono::milliseconds> tp;
    in >> date::parse("%FT%TZ", tp);
    if (in.fail())
    {
        in.clear();
        in.exceptions(std::ios::failbit);
        in.str(save);
        in >> date::parse("%FT%T%Ez", tp);
    }
    return tp;
}

int
main()
{
    using namespace date;
    using namespace std;
    cout << parse8601(istringstream{"2014-11-12T19:12:14.505Z"}) << '\n';
    cout << parse8601(istringstream{"2014-11-12T12:12:14.505-5:00"}) << '\n';
}

这个输出:

2014-11-12 19:12:14.505
2014-11-12 17:12:14.505

请注意,两个输出都是 UTC。 parse使用 -5:00 将本地时间转换为 UTC抵消。如果你真的想要本地时间,还有一种方法可以解析成一个叫做date::local_time<milliseconds>的类型。然后它将解析但忽略偏移量。甚至可以将偏移量解析为 chrono::minutes如果需要(使用 parse 重载以 minutes& )。

解析的精度由chrono::time_point 的精度控制。您传入,而不是通过格式字符串中的标志。并且偏移量可以是 +/-hhmm 的样式与 %z , 或 +/-[h]h:mm%Ez .

关于c++ - 如何在 C++ 中将 iso 8601 日期(可选毫秒)解析为 struct tm?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26895428/

相关文章:

C++ 分配仅具有已知列和未知行的矩阵时出现问题

php - 从文本中删除所有 HTML 标签+内容

c++ - 可移植打印和比较 pthread_t

c - fseek/fsetpos 可能会丢弃流缓冲区?

c - 扩展正则表达式是否支持反向引用?

python - 如果从源代码构建 BCC 后 "sudo/usr/share/bcc/tools/execsnoop"失败,我该怎么办?

c++ - Lint 更喜欢 std :string rather than clear() 上的 erase() 方法

c++ - 使用 "got there first"标志而不是锁/互斥来同步对数据的访问

parsing - powershell从命令输出中获取内容

git - 添加包含分号的 Git 别名