贝利信息

c++20的日历和时区库(std::chrono)如何使用? (处理时间日期)

日期:2026-01-13 00:00 / 作者:尼克
std::chrono::year_month_day 不能直接输出,因为它是一个纯数据结构,仅存储年、月、日三个整数字段,未重载 operator

std::chrono::year_month_day 为什么不能直接输出?

因为 std::chrono::year_month_day 是一个纯数据结构,不带格式化能力。它只存年、月、日三个整数字段,没有重载 operator,所以 std::cout 会编译失败。

std::chrono::year_month_day ymd{2025y/3/15};
std::chrono::sys_days sd{ymd}; // 必须这一步
std::cout << std::format("{:%Y-%m-%d}", sd); // 输出 "2024-03-15"

std::chrono::zoned_time 怎么正确绑定时区?

不能直接传字符串如 "Asia/Shanghai" 给构造函数 —— C++20 标准库不内置时区数据库,依赖系统实现(libstdc++ 和 libc++ 行为不同)。

try {
    const auto* tz = std::chrono::locate_zone("Asia/Shanghai");
    auto zt = std::chrono::zoned_time{tz, std::chrono::system_clock::now()};
} catch (const std::runtime_error& e) {
    // 例如 "Unknown time zone: Asia/Shanghai"
}

std::chrono::month_day_last 和 std::chrono::month_day 区别在哪?

它们都用于表示“某月的某日”,但语义和行为完全不同:

auto mdy = std::chrono::year_month_day{2025y / std::chrono::April / std::chrono::last};
std::cout << std::format("{:%F}", std::chrono::sys_days{mdy}); // "2024-04-30"

跨时区转换为什么容易漏掉夏令时(DST)?

因为 std::chrono::zoned_time 的转换逻辑完全依赖底层时区规则(TZDB),而 DST 规则每年可能变化。如果 TZDB 版本过旧,或系统未更新 zoneinfo 数据,zoned_time 会返回错误偏移。

真正可靠的跨时区操作,只应通过 zoned_time + get_sys_time() / get_local_time() 接口完成,不碰 raw offset。