登录
首页 >  Golang >  Go教程

Go定时器与ticker用法及任务详解

时间:2026-01-19 09:15:39 199浏览 收藏

知识点掌握了,还需要不断练习才能熟练运用。下面golang学习网给大家带来一个Golang开发实战,手把手教大家学习《Go中timer与ticker的使用及定时任务详解》,在实现功能的过程中也带大家重新温习相关知识点,温故而知新,回头看看说不定又有不一样的感悟!

Timer用于一次性定时,Ticker用于周期性定时;Timer的channel只发一次信号,Ticker则持续发送直至Stop;误用Ticker当Timer会导致逻辑错误。

Go并发编程中timer和ticker怎么用_定时并发任务说明

Go 里 time.Timertime.Ticker 不是“定时执行函数”的工具,它们只负责按时间发送信号——你得自己接住 channel 并处理逻辑,否则会漏触发、阻塞或 panic。

什么时候该用 Timer,什么时候用 Ticker

Timer 是一次性倒计时:启动后只向 C 字段的 chan time.Time 发一次信号;Ticker 是周期性节拍器:从启动起每隔固定间隔发一次信号,直到被显式 Stop()

  • 需要“5 秒后做一次清理” → 用 Timer
  • 需要“每 2 秒检查一次服务健康状态” → 用 Ticker
  • 误把 TickerTimer 用(比如只读一次 ),会导致 goroutine 泄漏 —— 它还在后台持续发信号,没人收
  • 反过来,用 Timer 做轮询,就得每次 Reset(),但要注意:如果旧 timer 还没触发就 Reset(),它会先 Stop() 再重启,不会重复触发

Timer 的常见误用和修复方式

最典型错误是忽略 Timer 的可重用性与生命周期管理。它不是“用完即焚”,但也不是线程安全的,不能并发调用 Reset()Stop()

  • 启动后不读 timer.C → 定时器触发后,time.Time 值会永久卡在 channel 中,下次 Reset()Stop() 可能 panic
  • 在 select 中直接写 case 而不先 timer.Reset() → 第二次定时永远不会开始
  • 并发调用 timer.Reset() → 可能 panic:“timer already fired” 或 “send on closed channel”
timer := time.NewTimer(3 * time.Second)
// ✅ 正确:select 中收信号,并在需要时 Reset
go func() {
    for {
        select {
        case <h3><code>Ticker</code> 的资源泄漏和停止时机</h3><p><code>Ticker</code> 启动后会持续往 <code>C</code> channel 发送时间值,**必须手动 <code>Stop()</code>**,否则 goroutine 和 channel 会一直存在,GC 不回收。</p>
  • for range ticker.C 循环中 break,但没调 ticker.Stop() → 泄漏
  • select 中监听 ticker.C,但程序退出前忘记 Stop() → 同样泄漏
  • ticker.Stop() 后继续读 ticker.C → 永远阻塞(channel 已关闭,但没人发值)
  • 不要用 time.AfterFunc() 替代 Ticker 做轮询:它是单次的,且内部用 Timer 实现,无法复用
ticker := time.NewTicker(1 * time.Second)
defer ticker.Stop() // 确保退出前停止
<p>go func() {
for {
select {
case t := <-ticker.C:
fmt.Printf("tick at %v\n", t)
}
}
}()</p>

并发任务中怎么避免时间精度丢失和堆积

Go 的 Timer/Ticker 底层依赖系统调度,**不保证绝对准时**。尤其在 GC STW、高负载或大量 goroutine 竞争时,可能延迟数百毫秒甚至更久。更严重的是:如果你的处理逻辑耗时 > 间隔,Ticker 会“攒着”多个未消费的 tick,导致后续集中爆发。

  • 对实时性要求高的场景(如游戏帧同步),别依赖 Ticker 的“节奏”,改用自适应 sleep:time.Sleep(nextTick.Sub(time.Now()))
  • select + default 非阻塞读 ticker.C,防止一次处理太久导致积压
  • 不要在 ticker.C 的接收逻辑里做阻塞 IO 或长耗时计算 —— 开新 goroutine 处理,但注意控制并发数
  • TimerReset() 在 timer 已触发后返回 false,记得判断返回值,否则可能误以为重置成功

真正难的不是调用 API,而是想清楚:这个定时信号是“提醒我该干活了”,还是“必须在这个时刻干完”。前者用 Ticker + 非阻塞处理,后者得结合上下文控制节奏,甚至放弃标准库 timer,改用基于 monotonic clock 的自定义调度器。

以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于Golang的相关知识,也可关注golang学习网公众号。

前往漫画官网入口并下载 ➜
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>