如何使用 GCC 9 将 `std::filesystem::file_time_type` 转换为字符串

ttt*_*apa 9 c++ std-filesystem gcc9

我正在尝试将文件的修改时间格式化为字符串 (UTC)。以下代码使用 GCC 8 编译,但不使用 GCC 9。

#include <chrono>
#include <filesystem>
#include <iomanip>
#include <iostream>
#include <sstream>

namespace fs = std::filesystem;

int main() {
    fs::file_time_type file_time = fs::last_write_time(__FILE__);
    std::time_t tt = decltype(file_time)::clock::to_time_t(file_time);
    std::tm *gmt = std::gmtime(&tt);
    std::stringstream buffer;
    buffer << std::put_time(gmt, "%A, %d %B %Y %H:%M");
    std::string formattedFileTime = buffer.str();
    std::cout << formattedFileTime << '\n';
}
Run Code Online (Sandbox Code Playgroud)

我使用 C++17 和 C++20 都尝试过decltype(file_time)::clockand std::chrono::file_clock,但它们都不起作用。

$ g++-9 -std=c++2a -Wall -Wextra -lstdc++fs file_time_type.cpp

file_time_type.cpp: In function ‘int main()’:
file_time_type.cpp:12:50: error: ‘to_time_t’ is not a member of ‘std::chrono::time_point<std::filesystem::__file_clock>::clock’ {aka ‘std::filesystem::__file_clock’}
   12 |     std::time_t tt = decltype(file_time)::clock::to_time_t(file_time);
      |                                                  ^~~~~~~~~
Run Code Online (Sandbox Code Playgroud)

https://en.cppreference.com/w/cpp/filesystem/file_time_type上的示例提到它不适用于 GCC 9,因为 C++20 将允许可移植输出,但我不知道如何获得它在职的。如果我不使用 C++20,它不应该与 GCC 9 一起使用吗?

如果可能,我更喜欢带有 GCC 9 的 C++17 解决方案。

Gul*_*rak 7

由于system_clockto_time_t,最简单的方法是转换到它。这并不完美(由于精度问题),但大多数情况下已经足够好,而且我在 MSVC 上也使用了:

template <typename TP>
std::time_t to_time_t(TP tp)
{
    using namespace std::chrono;
    auto sctp = time_point_cast<system_clock::duration>(tp - TP::clock::now()
              + system_clock::now());
    return system_clock::to_time_t(sctp);
}
Run Code Online (Sandbox Code Playgroud)

更新:由于我还不能发表评论,请澄清一下为什么会这样以及可能存在什么问题:它起作用是因为同一时钟的两个时间点之间的差异很容易,而对于第二部分,有一个额外模板operator+持续时间时间点不同的源(2)和比率的差异将采取的护理std::common_type

剩下的问题是,两个调用now()不是同时进行的,由于调用之间的时间差,存在引入小的转换错误的轻微风险。还有另一个关于 C++11 时钟转换的问题,它更详细地介绍了错误概率和减少错误的技巧,但是如果您不需要比较结果的往返转换,而只想格式化时间戳,这个较小的解决方案应该足够好了。

因此,要完成对原始问题的回答:

#include <chrono>
#include <filesystem>
#include <iomanip>
#include <iostream>
#include <sstream>

namespace fs = std::filesystem;

template <typename TP>
std::time_t to_time_t(TP tp)
{
    using namespace std::chrono;
    auto sctp = time_point_cast<system_clock::duration>(tp - TP::clock::now()
              + system_clock::now());
    return system_clock::to_time_t(sctp);
}

int main() {
    fs::file_time_type file_time = fs::last_write_time(__FILE__);
    std::time_t tt = to_time_t(file_time);
    std::tm *gmt = std::gmtime(&tt);
    std::stringstream buffer;
    buffer << std::put_time(gmt, "%A, %d %B %Y %H:%M");
    std::string formattedFileTime = buffer.str();
    std::cout << formattedFileTime << '\n';
}
Run Code Online (Sandbox Code Playgroud)