登录
首页 >  Golang >  Go教程

Golang定时器与Ticker并发使用方法

时间:2026-01-16 15:02:32 359浏览 收藏

一分耕耘,一分收获!既然都打开这篇《Golang定时器与Ticker并发使用技巧》,就坚持看下去,学下去吧!本文主要会给大家讲到等等知识点,如果大家对本文有好的建议或者看到有不足之处,非常欢迎大家积极提出!在后续文章我会继续更新Golang相关的内容,希望对大家都有所帮助!

timer.After 不能用于循环重置定时任务,因其返回单次通道、读完即关闭;应改用 time.NewTicker 或 time.NewTimer 配合 Reset。

Golang定时器与Ticker在并发中的应用

为什么 timer.After 不能直接用于循环重置定时任务

很多人想用 timer.After 实现“每 N 秒执行一次”,结果发现逻辑只跑了一次就停了。这是因为 timer.After 返回的是单次 chan,读完一次就关闭,无法复用。

正确做法是用 time.NewTimer 配合 Reset(),或更常见的——直接用 time.Ticker

  • time.After(d):适合「延迟一次」,比如超时控制、延后启动
  • time.NewTimer(d):适合「延迟一次 + 可手动重置」,比如带取消逻辑的单次任务
  • time.NewTicker(d):适合「周期性触发」,底层复用同一个定时器,比反复 new Timer 更轻量

在 goroutine 中使用 Ticker 必须显式 stop

不调用 ticker.Stop() 会导致 goroutine 和底层 ticker 持续运行,即使外层逻辑已退出。Go runtime 不会自动回收活跃的 time.Ticker,它会一直向 ticker.C 发送时间点,造成 goroutine 泄漏和内存缓慢增长。

典型错误写法:

go func() {
    ticker := time.NewTicker(5 * time.Second)
    for range ticker.C {
        doWork()
    }
}()

正确写法(带退出控制):

done := make(chan struct{})
go func() {
    defer close(done)
    ticker := time.NewTicker(5 * time.Second)
    defer ticker.Stop() // 关键:确保退出前释放资源
    for {
        select {
        case <h3>并发场景下 Ticker 与 channel select 的配合要点</h3><p>多个定时器或混合 I/O 事件时,<code>select</code> 是标准解法,但要注意几个易错点:</p>
  • 如果 ticker.C 和其它 channel 同时就绪,select 是伪随机选择,不能依赖顺序
  • 不要在 case 分支里做耗时操作,否则会拖慢下一次 tick —— Ticker 不跳过未消费的 tick,而是累积在 channel 缓冲区(默认缓冲 1),若持续阻塞,可能引发堆积或 panic(当缓冲满且无 receiver)
  • 如需严格节拍(如心跳、采样),应把耗时逻辑扔进新 goroutine,或用带缓冲的 channel 解耦

例如安全的节拍处理:

ticker := time.NewTicker(1 * time.Second)
defer ticker.Stop()
for {
    select {
    case <h3>Timer.Reset 和 Ticker.Reset 的行为差异</h3><p>两者都支持 <code>Reset(d)</code>,但语义不同:</p>
  • Timer.Reset(d):停止当前计时,重新以 d 开始倒计时;若原 timer 已触发,Reset 仍有效;若原 timer 未触发但已 Stop()Reset 会重新激活它
  • Ticker.Reset(d):立即停止当前 ticker,并以新间隔 d 创建一个全新 ticker;旧的 ticker.C channel 不再有值,必须用新的 channel(但通常没人这么干,因为 Reset 对 ticker 很少必要)

所以实际中:Timer.Reset 常见于动态超时调整;Ticker.Reset 几乎不用,改用 Stop() + NewTicker() 更清晰。

真正容易被忽略的是:所有 Reset 调用前,必须确保原 timer/ticker 没有处于已触发但尚未读取的状态,否则可能 panic 或行为异常 —— 尤其在多 goroutine 并发调用 Reset 时,务必加锁或通过 channel 协作。

好了,本文到此结束,带大家了解了《Golang定时器与Ticker并发使用方法》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多Golang知识!

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