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 各自的用途,避免混用类型。