c++20 的 std::chrono 引入 year_month_day 和 zoned_time 支持日历与时区处理,可通过 locate_zone 和 current_zone 获取时区信息,结合 format 实现 格式化输出 ,需注意平台对 IANA 数据库 的支持。

在 C ++20 中,std::chrono 进行了重大扩展,引入了对日历和时区的原生支持。这使得开发者无需依赖第三方库(如 Howard Hinnant 的 date 库或 Boost)就能处理日期、时间点与时区转换。以下是使用 C++20 的 std::chrono 处理日历和时区的核心方法。
1. 年 - 月 - 日 日历操作
C++20 引入了 year_month_day 类型,可以直接表示公历中的日期,并支持格式化与有效性检查。
示例:构造并验证一个日期
#include <chrono> #include <iostream> <p>int main() { using namespace std::chrono;</p><pre class="brush:php;toolbar:false;"><pre class="brush:php;toolbar:false;">// 构造日期 year_month_day ymd{year{2025}, month{3}, day{28}}; if (ymd.ok()) {std::cout << " 有效日期: " << ymd << 'n'; // 输出: 2025-03-28} else {std::cout << " 无效日期 n ";} // 从 time_point 转换为日历日期 auto now = system_clock::now(); auto today = floor<days>(now); year_month_day today_ymd = today; std::cout << " 今天是: " << today_ymd << 'n';
}
2. 时间点与时区转换
通过 zoned_time 可以将时间点与特定时区关联,并进行跨时区转换。
立即学习“C++ 免费学习笔记(深入)”;
示例:获取本地时间和不同时区的时间
#include <chrono> #include <iostream> <p>int main() { using namespace std::chrono;</p><pre class="brush:php;toolbar:false;"><pre class="brush:php;toolbar:false;">// 获取当前 UTC 时间 auto now = system_clock::now(); // 创建带时区的时间对象(自动使用系统默认时区)zoned_time local_time{current_zone(), now}; std::cout << " 本地时间: " << local_time << 'n'; // 转换为纽约时间 zoned_time ny_time{"America/New_York", now}; std::cout << " 纽约时间: " << ny_time << 'n'; // 转换为东京时间 zoned_time tokyo_time{"Asia/Tokyo", now}; std::cout << " 东京时间: " << tokyo_time << 'n';
}
3. 手动解析与时区映射
可以手动指定时区名称或偏移量来创建 zoned_time。IANA 时区数据库名称被广泛支持(需运行环境支持)。
常见用法:
- current_zone():获取程序运行时的本地时区
- locate_zone(“Zone/Name”):根据 IANA 名称查找时区
- sys_info / local_info:查询时区信息(如夏令时状态)
示例:查看时区信息
const time_zone* tz = locate_zone("Europe/London"); auto info = tz->get_info(sys_days{June/1/2025y}); std::cout << "UTC 偏移: " << info.offset << 'n'; std::cout << " 是否夏令时: " << (info.save != 0min) << 'n';
4. 格式化输出 日期时间
C++20 支持使用 std::format 或流输出直接格式化 chrono 类型。
示例:自定义格式输出
#include <format> <p>auto now = zoned_time{current_zone(), system_clock::now()}; std::string s = std::format("{:%Y 年 %m 月 %d 日 %H:%M:%S %Z}", now); std::cout << s << 'n'; // 如: 2025 年03 月 28 日 14:30:22 CST
注意:需要编译器支持 <format>(GCC 13+, Clang 14+, MSVC 2022)
基本上就这些。C++20 的 chrono 让日历和时区处理变得直观且安全。只要注意平台是否完整支持 IANA 时区数据库(linux 通常没问题,windows 可能需要补丁),就可以写出跨平台的时间逻辑。关键是理解 time_point、local_time、zoned_time 和 year_month_day 各自的用途,避免混用类型。