time.Ticker适用于固定间隔、持续运行的定时任务,需显式调用Stop避免资源泄漏,配合select可实现优雅退出,且不保证准时但保持均匀触发。

在 Go 中,time.Ticker 是实现周期性执行函数的常用方式,适合需要**固定间隔、持续运行**的定时任务(比如每 5 秒拉取一次状态、每分钟清理缓存等)。它比反复调用 time.AfterFunc 更简洁稳定,也比手动管理 time.Timer 更省心。
创建并启动 Ticker
使用 time.NewTicker(d) 创建一个以指定持续时间 d 周期触发的 ticker。它会立即返回一个 *time.Ticker,其 C 字段是一个只读的 chan time.Time,每次到点就会发送当前时间戳。
示例:每 2 秒执行一次打印
ticker := time.NewTicker(2 * time.Second) defer ticker.Stop() // 重要:避免 goroutine 和资源泄漏for range ticker.C { fmt.Println("执行任务:", time.Now().Format("15:04:05")) }
安全停止 Ticker(避免 Goroutine 泄漏)
如果不显式调用 ticker.Stop(),即使外层逻辑结束,ticker 仍会在后台持续发送时间事件,导致 goroutine 和定时器资源无法释放——这是常见陷阱。
立即学习“go语言免费学习笔记(深入)”;
- 在循环外提前定义 ticker,用
defer ticker.Stop()确保退出时关闭 - 若需动态控制启停(如根据条件暂停),用
select+done通道配合stop信号 - 不要在
for range ticker.C内部直接 break 后忽略 stop —— 必须显式调用ticker.Stop()
在 select 中结合其他通道(推荐生产用法)
实际项目中,往往需要让 ticker 与退出信号、错误通道、用户输入等协同工作。此时应把 ticker.C 放入 select 语句中,避免阻塞,并支持优雅退出。
done := make(chan bool) ticker := time.NewTicker(3 * time.Second) defer ticker.Stop()go func() { // 模拟某条件满足后通知退出 time.Sleep(10 * time.Second) done <- true }()
for { select { case t := <-ticker.C: fmt.Printf("任务执行于 %s\n", t.Format("15:04:05")) case <-done: fmt.Println("收到退出信号,停止 ticker") return } }
注意:Ticker 不保证“准时”,但保证“均匀”
time.Ticker 的底层基于系统时钟和调度器,如果任务执行耗时过长(比如单次耗时 > 间隔),后续 tick 会“堆积”或被跳过(Go runtime 会自动跳过已错过的 tick,不累积)。
- 例如间隔 1 秒,但某次任务跑了 3.2 秒,则接下来最多只触发 1 次(紧接在 3.2 秒后),不会补发第 2、3 次
- 若需严格按绝对时间点执行(如每整点触发),应改用
time.Timer循环重置,或借助外部调度器 - 对延迟敏感的任务,建议监控单次执行耗时,必要时加超时或降级
不复杂但容易忽略:Stop 要及时,select 要写全,耗时要量清。










