首页 > 解决方案 > std::time_point 从和到 std::string

问题描述

我正在尝试使用 c++20 std::chrono 替换一些 boost::gregorian 代码,希望消除 boost 构建依赖。代码正在读取和写入 json(使用 nlohmann),因此将日期与 std::string 相互转换的能力至关重要。

在 Ubuntu 20.04 上使用 g++ 9.3.0。2 个编译时错误,一个在 std::chrono::parse() 上,第二个在 std::put_time()

对于 std::chrono::parse() 上的错误 A,我在这里看到包含 chrono::parse 的日历支持 (P0355R7) 在 gcc libstdc++ 中尚不可用。任何人都知道这是否正确或有指向此 ETA 的链接?或者我调用 parse() 的方式有什么问题吗?

对于 std::put_time() 的错误 B:因为 std:put_time() 被记录为 c++11,感觉我在这里遗漏了一些愚蠢的东西。还发现需要通过 c 的 time_t 和 tm 进行隐藏很奇怪。有没有更好的方法可以将 std::chrono::time_point 直接转换为 std::string 而不求助于 c?

#include <chrono>
#include <string>
#include <sstream>
#include <iostream>

int main(int argc, char *argv[]) {
    std::chrono::system_clock::time_point myDate;

    //Create time point from string
    //Ref: https://en.cppreference.com/w/cpp/chrono/parse
    std::stringstream ss;
    ss << "2020-05-24";
    ss >> std::chrono::parse("%Y-%m-%e", myDate);   //error A: ‘parse’ is not a member of ‘std::chrono’

    //Write time point to string
    //https://en.cppreference.com/w/cpp/io/manip/put_time
    //http://cgi.cse.unsw.edu.au/~cs6771/cppreference/en/cpp/chrono/time_point.html
    std::string dateString;
    std::time_t dateTime = std::chrono::system_clock::to_time_t(myDate);
    std::tm tm = *std::localtime(&dateTime);
    dateString = std::put_time(&tm, "%Y-%m-%e");    //error B: ‘put_time’ is not a member of ‘std’

    //Write out
    std::cout << "date: " << dateString << "\n";

    return 0;
}

标签: c++c++20chrono

解决方案


用于 gcc 的C++20<chrono>仍在构建中。我没有看到任何公开的预计到达时间。

你的语法std::chrono::parse看起来是正确的。如果您愿意使用免费的、开源的、仅包含标头的 C++20 预览版,<chrono>#include "date/date.h"那么您可以通过添加和使用来使其工作date::parse

请注意,结果myDate将是 2020-05-24 00:00:00 UTC。

std::put_time生活在头部<iomanip>,是一个操纵者。添加该标题后,<iostream>您将像这样使用它:

std::cout << "date: " << std::put_time(&tm, "%Y-%m-%e") << '\n';

如果您需要 a 中的输出std::string,则必须将操纵器流式传输到std::stringstream第一个。

C++20<chrono>将提供 C API 的替代方案来进行格式化:

std::cout << "date: " << std::format("{%Y-%m-%e}", myDate) << '\n';

预览库还提供了一个稍微改变的格式字符串:

std::cout << "date: " << date::format("%Y-%m-%e", myDate) << '\n';

推荐阅读