std::chrono::system_clock::now() 获取自1970-01-01 UTC起的纳秒级时间点,需用 to_time_t() 转为 time_t 后才能通过 ctime 等函数输出可读时间。

用 std::chrono::system_clock 获取当前时间点
std::chrono::system_clock 是 C++11 起标准库中表示系统实时时钟的类型,它返回的是自纪元(通常是 1970-01-01 00:00:00 UTC)以来的纳秒级时间点。直接调用 now() 即可拿到当前时间点:
auto now = std::chrono::system_clock::now();
注意:返回值是 std::chrono::time_point<system_clock, duration> 类型,不能直接打印或比较,需转为 time_t 或字符串才能使用。
转成 time_t 再用 std::ctime 输出可读时间
大多数传统时间函数(如 std::localtime、std::gmtime)只接受 time_t,所以得先用 to_time_t() 转换:
auto now = std::chrono::system_clock::now(); std::time_t t = std::chrono::system_clock::to_time_t(now); std::cout << std::ctime(&t); // 输出类似 "Wed Jun 12 15:23:45 2024\n"
常见错误:直接对 now 调用 std::ctime 会编译失败,因为参数类型不匹配。
立即学习“C++免费学习笔记(深入)”;
-
std::ctime返回带换行符的 C 风格字符串,结尾有\n - 若需去掉换行,可用
std::string(ctime(&t)).c_str()截断,或用std::put_time更精细控制格式 -
to_time_t()会截断到秒级精度,丢失毫秒/微秒信息
用 std::put_time 格式化输出(推荐)
比起 std::ctime 的固定格式,std::put_time 支持自定义格式,且能保留 system_clock::to_time_t 转换后的秒级时间(再配合 duration_cast 补毫秒):
auto now = std::chrono::system_clock::now();
auto time_t = std::chrono::system_clock::to_time_t(now);
auto ms = std::chrono::duration_cast<std::chrono::milliseconds>(now.time_since_epoch()) % 1000;
std::cout << std::put_time(std::localtime(&time_t), "%Y-%m-%d %H:%M:%S");
std::cout << '.' << std::setfill('0') << std::setw(3) << ms.count() << '\n';
关键点:
- 必须包含
<iomanip>和<chrono> -
std::localtime不是线程安全的,多线程下建议用std::localtime_r(POSIX)或封装成线程局部变量 - 上面示例中毫秒是“纪元后总毫秒数对 1000 取余”,等价于当前秒内的毫秒偏移
跨平台注意事项和精度陷阱
system_clock 在不同平台实现不同:Linux/macOS 通常基于 clock_gettime(CLOCK_REALTIME),Windows 上早期 MSVC 实现精度只有 15ms 左右(VS2015 后已改善)。不要假设它一定能返回毫秒或微秒级稳定精度。
- 若需高精度单调时钟(比如测耗时),应改用
std::chrono::steady_clock,它不随系统时间调整而跳变 -
system_clock::now()可能因 NTP 同步、手动调时而回退或跳跃,不适合做超时等待逻辑 - UTC 时间转换要小心:用
std::gmtime替代std::localtime,但同样存在线程安全问题
真正需要精确到毫秒的日志时间戳,建议统一用 system_clock::now() 存储原始 time_point,只在最终展示时格式化——避免反复转换引入误差。










