Go 的时间包:理解单调时间与挂钟时间

Go 的时间包:理解单调时间与挂钟时间 📅

引言

Go 语言自版本 1.9 起在 time.Time 中同时支持 “挂钟时间(wall‑clock)” 和 “单调时间(monotonic clock)”,用于分别满足时间戳与时间间隔测量的需求。本文将帮你快速掌握二者区别与使用中的常见坑。


一、挂钟 vs 单调时间:何时何用?

  • 挂钟时间(wall‑clock):现实世界的时间,如 UTC 或本地时区,用于生成具有全局意义的时间戳,但可能因 NTP 校准、夏令时切换、闰秒插入等原因跳变 X (formerly Twitter)+2VictoriaMetrics+2X (formerly Twitter)+2。

  • 单调时间(monotonic):稳步前进、永不后退,专用于测量时间间隔,不受系统时间调整影响 VictoriaMetrics。

在 Go 中,这两者各司其职:挂钟用于“告诉我现在几点”,单调时间用于“测量经过了多久”。


二、Go 中的实现方式

  • time.Now() 返回的 time.Time 结构内同时包含壁钟与单调时间组件(后者存储在 ext 字段的隐藏部分) VictoriaMetrics。

  • 只有通过 runtime 调用的 time.Now() 和相关函数才带有单调时间。其他构造方式如 time.Date, time.Parse, time.Unix 不包含单调时间,只保留挂钟部分 VictoriaMetrics。

运行时输出的 m=+0.xxx 后缀即表示自程序启动以来的单调时间偏移,例如:

 

yaml

複製編輯

2025‑07‑25 … m=+0.000123456

说明这段时间使用的是单调时间测量 VictoriaMetrics。


三、常见误区与正确比较方式

❌ 比较 time.Time==
  • 直接用 == 比较可能错误,因为 UTC()Truncate() 会清除或重设内部单调时间或位置指针,导致相等时间比较结果为 false VictoriaMetrics。

应使用:

go

複製編輯

t1.Equal(t2)

它会根据是否同时拥有单调时间来判断:若都有,则比较 ext;否则比较挂钟时间(秒与纳秒) X (formerly Twitter)+2VictoriaMetrics+2X (formerly Twitter)+2。


四、测量时间间隔的正确方式

  • 推荐使用带单调时间的 time.Since(t) 或者 time.Now().Sub(t),如果 t 拥有单调时间读取,则将使用单调方式计算间隔,避免挂钟跳变造成计算错误 VictoriaMetrics。

  • t 是通过 time.Parse 等构造的无单调信息,则会退回到挂钟计算,可能受到系统校时影响。

  • 性能优化技巧:在高频路径下,可以使用:

     

    go

    複製編輯

    past := time.Now() // ... nowApprox := past.Add(time.Since(past))

    可提升 ~50% 性能,但不再捕捉系统时间调整。如果你不需要对校时敏感,可适用此技巧 VictoriaMetrics。


五、计划调度中的壁钟风险

  • 对于基于“现在是几点?”或“下一次调度是什么时间?”的应用(如 cron 作业、日志轮转、告警检查等),应使用挂钟时间进行比较,即使挂钟可能跳变。

  • monotinic 时钟在系统挂起(sleep)后通常会暂停,这可能导致间隔计算不含挂起时间,所以调度应当依赖壁钟路径 VictoriaMetrics。


六、底层结构揭秘

  • Go 1.9 之前,time.Time 的结构体只有 sec, nsec, loc,不支持单调时间。 Go 1.9 开始引入 wall uint64, ext int64, loc *Location 三字段形式,在 24 字节内兼容保存挂钟与单调时间信息 VictoriaMetrics。

  • 当只存挂钟时,ext 用作 wall 秒字段;有单调时间时,wall 秒搬移进 wallext 变为 monotonic 时间读取。

  • 如果时间运算(如 Add())导致 monotonic 范围超出,Go 会自动剥离 monotonic 信息,只保留壁钟时间 VictoriaMetrics。


总结建议

用途推荐方式
获取当前时间戳time.Now()(包含挂钟)
测量时间间隔time.Since(t)time.Now().Sub(t)(使用单调时间)
高频日期时间计算(无需 sync 调整)past.Add(time.Since(past))
基于日历时间的调度任务使用挂钟比较(保留跳变行为)


结语

理解 Go 的挂钟与单调时间机制,对于写出健壮且高性能的时间处理逻辑至关重要。特别是在涉及延迟测量、调度任务或系统校时情况时,这项知识能避免许多 subtle bug。

原文作者 Phuong Le 于 2025 年 7 月 25 日发布的这篇文章深入技术细节,对 Go 时间包底层实现给出了清晰讲解,非常值得一读 VictoriaMetrics+4VictoriaMetrics+4VictoriaMetrics+4。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/web/90893.shtml
繁体地址,请注明出处:http://hk.pswp.cn/web/90893.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Android启动时间优化大全

1 修改Android mksh默认的列长度 不修改这个参数,adb shell后,输入超过80个字符,就不能看到完整的命令行。external/mksh/src/sh.h EXTERN mksh_ari_t x_cols E_INIT(80); EXTERN mksh_ari_t x_lins E_INIT(24);2 Kernel优化 2.1 内核驱动模块…

matplotlib.pyplot: 底层原理简析与进阶技巧

文章目录 1 底层实现原理 1.1 核心架构 1.1 渲染流程 2 基础用法 2.1 基本绘图 2.2 多子图系统 2.3 高阶用法 2.3.1 自定义Artist对象 2.3.2 高级动画技术 2.3.3 事件处理系统 2.3.4 混合渲染技术 3 性能优化技巧 4 扩展模块 5 总结 5.1 底层原理关键点 5.2 进阶技巧 1 底层实现…

深入理解现代前端开发中的 <script type=“module“> 与构建工具实践

引言:模块化开发的演进在早期的前端开发中,JavaScript 缺乏原生的模块化支持,开发者不得不依赖 IIFE(立即调用函数表达式)或第三方库(如 RequireJS)来实现代码组织。随着 ES6(ES2015…

yolo--qt可视化开发

qt5可能不支持我们的cuda版本,改用qt6 YOLO11QT6OpencvC训练加载模型全过程讲解_yolov11 模型转换成opencv c模型-CSDN博客 下面是qt5版本的案例,和yolo及cuda有冲突 安装qt 切换到虚拟环境,例如pyqt,conda activate pyqt pip …

SQL性能优化

show [session|global] status : 查看服务器状态 show global status like Com_ : 查看各种语句的执行次数 开启慢查询: 在 MySQL 配置文件(/etc/my.cnf)配置: #开启MySQL慢日志查询开关 slow_query_log1 #设置慢日志的时间为2秒,SQL语句执…

ctfshow pwn40

目录 1. 分析程序 2. 漏洞编写 3. 漏洞验证 1. 分析程序 首先检查程序相关保护,发现程序为32位且只开启了一个NX保护 checksec pwn 使用IDA进行逆向分析代码,查看漏洞触发点: 在main函数中,有一个ctfshow函数,这里…

SQL173 店铺901国庆期间的7日动销率和滞销率

SQL173 店铺901国庆期间的7日动销率和滞销率 SQL题解:店铺动销率与滞销率计算 关键:只要当天任一店铺有任何商品的销量就输出该天的结果,即使店铺901当天的动销率为0。 潜台词:​输出逻辑与店铺901的销售情况无关,只取…

PytorchLightning最佳实践基础篇

PyTorch Lightning(简称 PL)是一个建立在 PyTorch 之上的高层框架,核心目标是剥离工程代码与研究逻辑,让研究者专注于模型设计和实验思路,而非训练循环、分布式配置、日志管理等重复性工程工作。本文从基础到进阶&…

Apache Flink 实时流处理性能优化实践指南

Apache Flink 实时流处理性能优化实践指南 随着大数据和实时计算需求不断增长,Apache Flink 已经成为主流的流处理引擎。然而,在生产环境中,高并发、大吞吐量和低延迟的业务场景对 Flink 作业的性能提出了更高要求。本文将从原理层面深入解析…

ubuntu上将TempMonitor加入开机自动运行的方法

1.新建一个TempMonitor.sh文件,内容如下:#!/bin/bashcd /fjrobot/ ./TempMonitor &2.执行以下命令chmod x TempMonitor chmod x TempMonitor.sh rm -rf /etc/rc2.d/S56TempMonitor rm -rf /etc/init.d/TempMonitor cp /fjrobot/TempMonitor.sh /etc/…

速卖通自养号测评技术解析:IP、浏览器与风控规避的实战方案

一、速卖通的“春天”来了,卖家如何抓住机会?2025年的夏天,速卖通的风头正劲。从沙特市场跃升为第二大电商平台,到8月大促返佣力度升级,平台对优质商家的扶持政策越来越清晰。但与此同时,竞争也愈发激烈——…

adb: CreateProcessW failed: 系统找不到指定的文件

具体错误 adb devices * daemon not running; starting now at tcp:5037 adb: CreateProcessW failed: 系统找不到指定的文件。 (2) * failed to start daemon adb.exe: failed to check server version: cannot connect to daemon 下载最新的platform-tools-windows 下载最新…

Centos安装HAProxy搭建Mysql高可用集群负载均衡

接上文MYSQL高可用集群搭建–docker https://blog.csdn.net/weixin_43914685/article/details/149647589?spm1001.2014.3001.5501 连接到你搭建的 Percona XtraDB Cluster (PXC) 数据库集群,实现高可用性和负载均衡,建议使用一个中间件来管理这些连接。…

Sql server开挂的OPENJSON

以前一直用sql server2008,自从升级成sql server2019后,用OPENJSON的感觉像开挂,想想以前表作为参数传输时的痛苦,不堪回首。一》不堪回首 为了执行效率,很多时候希望将表作为参数传给数据库的存储过程。存储过程支持自…

【数据结构】队列和栈练习

1.用队列实现栈 225. 用队列实现栈 - 力扣(LeetCode) typedef int QDatatype; typedef struct QueueNode {struct QueueNode *next;QDatatype data; }QNode;typedef struct Queue {QNode* head;QNode* tail;QDatatype size; }Que;typedef struct {Que…

LabVIEW二维码实时识别

​LabVIEW通过机器视觉技术,集成适配硬件构建二维码实时识别系统。通过图像采集、预处理、定位及识别全流程自动化,解决复杂环境下二维码识别效率低、准确率不足问题,满足工业产线追溯、物流分拣等实时识别需求。应用场景适用于工业产线追溯&…

微服务-springcloud-springboot-Skywalking详解(下载安装)

一、SkyWalking核心介绍 1. 什么是SkyWalking? Apache SkyWalking是一款国人主导开发的开源APM(应用性能管理)系统,2015年由吴晟创建,2017年进入Apache孵化器,2019年毕业成为Apache顶级项目。它通过分布式…

Elasticsearch 字段值过长导致索引报错问题排查与解决经验总结

在最近使用 Elasticsearch 的过程中,我遇到了一个 字段值过长导致索引失败 的问题。经过排查和多次尝试,最终通过设置字段 "index": false 方式解决。本文将从问题现象、排查过程、问题分析、解决方案和建议等方面,详细记录这次踩坑…

使用idea 将一个git分支的部分记录合并到git另一个分支

场景: 有多个版本分支,需要将其中一个分支的某一两次提交合并到指定分支上 eg: 将v1.0.0分支中指定提交记录 合并到 v1.0.1分支中 操作: 步骤一 idea切换项目分支到v1.0.1(需要合并到哪个分支就先站到哪个分支上) 步骤二 在ide…

基于深度学习的图像分类:使用ShuffleNet实现高效分类

前言 图像分类是计算机视觉领域中的一个基础任务,其目标是将输入的图像分配到预定义的类别中。近年来,深度学习技术,尤其是卷积神经网络(CNN),在图像分类任务中取得了显著的进展。ShuffleNet是一种轻量级的…