游乐游手机版
首页/编程语言/文章详情

C++如何获取文件创建时间 _ filesystem::last_write_time【干货】

时间:2026-05-05 13:04
C++如何获取文件创建时间:绕过filesystem限制的实用指南 std::filesystem::last_write_time 仅提供修改时间,无法获取创建时间 对于需要在Windows平台上使用C++获取文件创建时间的开发者而言,标准库可能会让你感到困惑。尽管C++20的std::files

C++如何获取文件创建时间:绕过filesystem限制的实用指南

C++如何获取文件创建时间 _ filesystem::last_write_time【干货】

std::filesystem::last_write_time 仅提供修改时间,无法获取创建时间

对于需要在Windows平台上使用C++获取文件创建时间的开发者而言,标准库可能会让你感到困惑。尽管C++20的std::filesystem模块提供了last_write_time(最后修改时间)和last_access_time(最后访问时间)函数,但它确实缺少一个标准的creation_time接口。这并非代码实现问题,而是当前C++标准库本身的设计局限。

请注意一个常见误区:部分开发者可能在MSVC的头文件中发现一些未公开的、名称类似__creation_time的成员,或在网络上找到某些非标准的“增强”实现。强烈建议避免使用这些方法,因为它们属于未定义行为,可能导致编译错误或在运行时引发难以预料的问题。

Windows平台解决方案:使用GetFileTime API与FILETIME转换

那么在Windows系统中,如何准确获取文件的创建时间呢?关键在于直接调用Windows原生API,绕过标准库的限制。核心步骤明确:首先通过CreateFileW打开文件并获得句柄,然后使用GetFileTime函数获取时间信息,最后进行必要的数据格式转换。

实现过程中需要关注以下技术细节:

  • GetFileTime函数会同时返回三个时间戳:创建时间(lpCreationTime)、最后访问时间(lpLastAccessTime)以及最后修改时间(lpLastWriteTime),调用时需确保参数顺序正确。
  • 该API返回的FILETIME结构体,其时间单位是100纳秒,计时起点为1601年1月1日(UTC)。这与常用的time_t(通常以秒为单位,起点为1970年)存在显著差异,不能直接互换使用。
  • 若希望返回的时间类型与std::filesystem::last_write_time保持一致(即std::chrono::file_clock::time_point),则需要进行转换。通常可使用std::chrono::file_clock::from_sys,但需注意Windows平台上file_clock的实现可能存在精度差异。
  • 完善的错误处理至关重要:务必检查CreateFileW返回的句柄是否为INVALID_HANDLE_VALUE,并确认GetFileTime调用返回TRUE。否则,获取到的时间数据将是无效的。

跨平台兼容性策略:避免强求创建时间,采用降级方案

当代码需要跨平台运行时,获取文件创建时间将面临更大挑战。Linux和macOS系统遵循的POSIX标准并未明确定义“文件创建时间”这一属性。尽管ext4、xfs等文件系统内部可能包含类似crtime的字段,APFS也记录创建时间,但这些信息通常不通过标准接口暴露,且行为不一致。标准的stat结构体仅包含st_mtime(修改时间)、st_atime(访问时间)和st_ctime(状态变更时间,请注意这并非创建时间)。

因此,若尝试封装一个通用的、跨平台的creation_time函数,很可能在非Windows平台上返回错误值或引发异常。

更可行的工程实践方案如下:

  • 首先评估业务需求:是否必须依赖文件的创建时间?例如用于日志审计或合规性检查。如果不是,那么使用last_write_time(最后修改时间)通常是更安全、兼容性更广的选择。
  • 若确实需要支持,应实现平台相关函数:在Windows上调用GetFileTime,在其他平台(如Linux、macOS)上则回退(fallback)到返回last_write_time,并添加明确注释说明该返回值“并非真实的创建时间”。
  • 在编写模板或泛型代码时需格外谨慎,切勿默认假设creation_time接口存在,否则代码可能在Linux CI/CD流水线中编译失败或运行时崩溃。

完整的Windows文件创建时间读取示例代码

#include 
#include 
#include 

std::optional get_creation_time(const std::filesystem::path& p) {
    HANDLE h = CreateFileW(p.c_str(), GENERIC_READ, FILE_SHARE_READ,
                          nullptr, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, nullptr);
    if (h == INVALID_HANDLE_VALUE) return std::nullopt;

    FILETIME ft_create, ft_access, ft_write;
    if (!GetFileTime(h, &ft_create, &ft_access, &ft_write)) {
        CloseHandle(h);
        return std::nullopt;
    }
    CloseHandle(h);

    // FILETIME → uint64_t → 100ns ticks since 1601 → convert to file_clock
    uint64_t ft_int = (static_cast(ft_create.dwHighDateTime) << 32) |
                      ft_create.dwLowDateTime;

    // Windows file_clock epoch is 1601-01-01 UTC, same as FILETIME
    auto tp = std::chrono::file_clock::time_point(
        std::chrono::duration_cast(
            std::chrono::nanoseconds(ft_int * 100)
        )
    );
    return tp;
}

需要说明的是,在MSVC编译器中,std::chrono::file_clock已映射到Windows的FILETIME时间体系。然而,使用GCC或Clang在Windows上编译时,其实现可能不完整。此时,可考虑改用system_clock并处理时区转换,或直接返回FILETIME原始值,由上层调用方决定如何处理。

最后强调一个关键但常被忽略的事实:在NTFS文件系统上,文件的创建时间并非绝对不可变。当文件被复制、移动或从压缩包解压时,该时间戳可能会被重置——具体行为取决于执行操作的工具。因此,切勿将创建时间视为唯一、具有强语义保证的元数据来使用。

来源:https://www.php.cn/faq/2344127.html
上一篇c#如何实现分布式事务_c#分布式事务最全用法总结 下一篇C#怎么使用Range和Index_C#范围和索引运算符详解教程【技巧】
本站内容用于信息整理与展示,如有侵权或内容问题请及时联系处理。

相关推荐

补充同频道和同主题内容,方便继续浏览更多相关内容。

同类最新

继续查看同栏目最近更新的文章。

更多
PyTorch中使用多维索引张量对高维张量批量索引的正确方法
编程语言 · 2026-07-03

PyTorch中使用多维索引张量对高维张量批量索引的正确方法

本文深入讲解如何在 PyTorch 中利用形状为 [b, k] 的索引张量 B,对形状为 [b, m, n] 的高维张量 A 执行高效批量索引,最终得到 [b, k, n] 的输出。核心思路在于合理扩展索引维度并配合 torch gather 实现精准的逐行抽取。 很多人处理高维张量的批量索引时都会

Go中...操作符解包切片传递可变参数函数
编程语言 · 2026-07-03

Go中...操作符解包切片传递可变参数函数

在 Go 语言中,` ` 运算符放在切片变量后面(如 `slice `)的作用是将该切片“展开”为多个独立参数,专门用于调用那些接受可变参数(` T`)的函数,例如 `append` 或 `fmt Println`。这是一种类型安全的语法糖,并非省略号或通配符,能够帮助开发者更简洁地处理

macOS与WSL2下PHP多版本切换失效问题排查与修复指南
编程语言 · 2026-07-03

macOS与WSL2下PHP多版本切换失效问题排查与修复指南

本文深入分析在 macOS 或 WSL2(Ubuntu)开发环境中,通过 Homebrew 管理 PHP 多版本时,php -v 始终显示旧版本(如 php@5 6)的深层原因,并给出系统性解决方案,覆盖 PATH 冲突、符号链接逻辑、Shell 初始化配置、系统残留配置等关键环节。 遇到这种情况的

PHP JSON解析深层嵌套对象属性访问失败的解决方法
编程语言 · 2026-07-03

PHP JSON解析深层嵌套对象属性访问失败的解决方法

使用 json_decode() 解析 API 返回的 JSON 数据时,经常遇到某个子属性无法正常获取,始终返回 NULL —— 这是许多 PHP 开发者都曾碰到过的棘手问题。通常并非数据丢失,而是对象嵌套层级比预期更深,导致访问路径不正确。 举例来说,你看到返回的 JSON 里有一个 appea

nnU-Net v2预处理卡死问题的成因分析与实用解决指南
编程语言 · 2026-07-03

nnU-Net v2预处理卡死问题的成因分析与实用解决指南

> 使用 nnUNetv2_plan_and_preprocess 处理大规模数据集(例如 704 例样本)时,程序常因多进程加载导致死锁而停滞。核心原因在于默认并发数过高引发资源竞争或 I O 阻塞,适当降低并发数即可稳定完成全量预处理。 你在使用 `nnunetv2_plan_and_prepr