news 2026/6/9 22:45:43

Go进阶之定时器Ticker

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Go进阶之定时器Ticker

Ticker是周期性定时器.即周期性的触发一个事件.通过Ticker本身提供的管道将事件

传递出去.

Ticker数据结构:

源码位置:src/time/tick.go

type Ticker struct { C <-chan Time // The channel on which the ticks are delivered. initTicker bool }

Ticker对外仅暴露了一个channel.当指定时间到来时就往该channel中写入系统时

间.即一个事件.

1.简单定时任务:

func main() { ticker := time.NewTicker(1 * time.Second) defer ticker.Stop() for range ticker.C { log.Println("tick") } }

执行结果:

定时聚合任务:

示例:

func TickerLaunch() { //5分钟执行一次. ticker := time.NewTicker(5 * time.Minute) defer ticker.Stop() maxPassenger := 30 passengers := make([]string, 0, maxPassenger) for { passenger := GetNewPassenger() if passenger != "" { passengers = append(passengers, passenger) } else { time.Sleep(1 * time.Second) } select { case <-ticker.C: Launch(passengers) passengers = []string{} default: if len(passengers) >= maxPassenger { Launch(passengers) passengers = []string{} } } } } func Launch(passengers []string) { for i := range passengers { fmt.Println(passengers[i], "发车了") } } func GetNewPassenger() string { return "乘客" }

2.Ticker对外接口:

1)创建定时器:

源码位置:src/time/tick.go

func NewTicker(d Duration) *Ticker { if d <= 0 { panic("non-positive interval for NewTicker") } // Give the channel a 1-element time buffer. // If the client falls behind while reading, we drop ticks // on the floor until the client catches up. c := make(chan Time, 1) t := (*Ticker)(unsafe.Pointer(newTimer(when(d), int64(d), sendTime, c, syncTimer(c)))) t.C = c return t }

2).停止定时器:

源码位置:src/time/tick.go

// Stop turns off a ticker. After Stop, no more ticks will be sent. // Stop does not close the channel, to prevent a concurrent goroutine // reading from the channel from seeing an erroneous "tick". func (t *Ticker) Stop() { if !t.initTicker { // This is misuse, and the same for time.Timer would panic, // but this didn't always panic, and we keep it not panicking // to avoid breaking old programs. See issue 21874. return } stopTimer((*Timer)(unsafe.Pointer(t))) }

流程图:

总流程图:

3.简单接口:

在有些场景下.启动一个定时器后.该定时器永远不会停止.比如定时轮询任务.此时可

以使用一个简单的Tick函数来获取定时器的管道.函数如下:

源码位置:src/runtime/tick.go

func Tick(d Duration) <-chan Time { if d <= 0 { return nil } return NewTicker(d).C }

从源码可以看出其实是创建了一个Ticker.然后只返回了管道.并没有返回ticker.所以

没有办法停止.

4.错误示例:

当Ticker用于for循环时.很容易出现意想不到的资源泄漏问题.

示例:

func WrongTicker() { for { select { case <-time.Tick(time.Second): log.Printf("resource leak!") } } }

上面的代码.select每次检测case语句都会创建一个定时器.for循环又会不断的执行

select语句.系统里会有越来越多的定时器不断地消耗CPU资源.最终CPU资源被耗

尽.

兜兜转转.何以解忧.

如果大家喜欢我的分享的话.可以关注我的微信公众号

念何架构之路

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/9 19:48:51

Qwen3与计算机网络:分布式字幕处理系统架构设计

Qwen3与计算机网络&#xff1a;分布式字幕处理系统架构设计 当视频内容以指数级增长&#xff0c;传统字幕处理方案已难以应对海量并发需求。本文将探讨如何基于计算机网络原理构建分布式Qwen3智能字幕处理系统&#xff0c;实现高效、可靠的大规模视频处理。 1. 系统架构概览 分…

作者头像 李华
网站建设 2026/6/9 21:04:57

STM32复位启动与中断向量表原理深度解析

1. STM32复位启动与中断控制原理深度解析嵌入式系统上电或复位后的第一行代码从何而来&#xff1f;中断发生时&#xff0c;CPU如何在毫秒级甚至微秒级内完成上下文切换并精准跳转到对应的服务函数&#xff1f;这些问题的答案&#xff0c;深植于ARM Cortex-M3内核的启动机制与异…

作者头像 李华
网站建设 2026/6/9 19:44:49

DAMO-YOLO-S模型知识蒸馏损失函数:KL散度与温度系数调优

DAMO-YOLO-S模型知识蒸馏损失函数&#xff1a;KL散度与温度系数调优 1. 引言 你有没有想过&#xff0c;为什么一个在服务器上跑得飞快的AI模型&#xff0c;一到手机上就变得又慢又耗电&#xff1f;这背后其实是一个经典的“大模型”与“小设备”的矛盾。大模型能力强&#xf…

作者头像 李华
网站建设 2026/6/9 21:10:25

BGE Reranker-v2-m3低代码集成方案:无需编程的API调用指南

BGE Reranker-v2-m3低代码集成方案&#xff1a;无需编程的API调用指南 1. 为什么你需要这个指南 你可能已经听说过BGE Reranker-v2-m3——这个由北京智源人工智能研究院开发的轻量级重排序模型&#xff0c;能精准判断查询和文档之间的相关性&#xff0c;让搜索结果更准确、问…

作者头像 李华
网站建设 2026/6/9 21:05:05

PasteMD异常处理机制:构建高可用的文档转换服务

PasteMD异常处理机制&#xff1a;构建高可用的文档转换服务 每次从AI对话里复制一大段内容&#xff0c;满怀期待地按下粘贴键&#xff0c;结果Word里一片乱码——公式变成天书&#xff0c;表格挤成一团&#xff0c;那种感觉就像精心准备的礼物在最后一刻摔碎了。作为经常和文档…

作者头像 李华