我有一个字符串,应该在 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/