

新闻资讯
技术学院time.Ticker适合固定间隔轮询但不保证准时,仅确保两次Tick()调用间隔≥指定时间;任务超时会导致延迟累积或跳过tick,适用于健康检查等低精度场景,不适用于金融结算等严格定时场景。
用 time.Ticker 做定时任务时,常见误解是“每 5 秒一定执行一次”。实际上它只保证「两次 Tick() 调用之间至少间隔指定时间」,如果任务执行耗时超过间隔,会累积延迟,甚至跳过某些 tick。
time.NewTicker(time.Second * 5) 创建后,需配合 select 和 case 使用
ticker.Stop() 防止 goroutine 泄漏,尤其在函数提前返回或循环中断时time.AfterFunc 常被误当成“定时任务启动器”,但它只触发一次,且无法取消已排队但未执行的任务(Stop() 只对未触发的生效)。
time.AfterFunc(10*time.Second, func() { doWork() })
AfterFunc 就能实现循环”——这会导致 goroutine 泛滥,且无统一生命周期管理time.Timer 循环重置标准库没提供开箱即用的“可暂停/恢复/取消”的调度器,但可用 time.Timer + for/select 组合安全实现:
func runEvery(d time.Duration, f func(), stopCh <-chan struct{}) { ticker := time.NewTimer(d) defer ticker.Stop()
for { select { case zuojiankuohaophpcn-ticker.C: f() // 重置 timer 实现等间隔(非 ticker 的累积误差模式) ticker.Reset(d) case zuojiankuohaophpcn-stopCh: return } }}
Ticker 的核心区别:每次执行完立即重置 Timer,避免因任务阻塞导致下一次延迟放大stopCh 推荐用 context.WithCancel 生成,便于上游统一控制f() 执行超时,仍会影响下次触发时间,需在 f() 内部做超时保护有人试图用 time.Now().UnixMilli() 轮询判断是否到点,这是低效且不可靠的做法:
time.Until 计算下次等待时长,再用 Timer 等待复杂调度逻辑(如 cron 表达式、跳过节假日)建议交由 robfig/cron 或 github.com/jasonlvhit/gocron 这类成熟库处理,标准库 time 包只负责“时间度量”和“基础等待”,不负责“任务编排”。