golang一次性定时器Timer用法及实现原理详解
来源:脚本之家
时间:2023-01-07 12:07:14 335浏览 收藏
本篇文章给大家分享《golang一次性定时器Timer用法及实现原理详解》,覆盖了Golang的常见基础知识,其实一个语言的全部知识点一篇文章是不可能说完的,但希望通过这些问题,让读者对自己的掌握程度有一定的认识(B 数),从而弥补自己的不足,更好的掌握它。
Timer
Timer
是一种单一事件的定时器,即经过指定的时间后触发一个事件,因为Timer
只执行一次就结束,所以称为单一事件,这个事件通过其本身提供的channel
进行通知触发。
timer结构体
通过src/time.sleep.go:Timer
定义了Timer
数据结构:
// Timer代表一次定时,时间到达后仅执行一个事件。 type Timer struct { C它提供了一个
channel
,在定时时间到达之前,没有数据写入timer.C
会一直阻塞,直到时间到达,向channel
写入系统时间,阻塞解除,可以从中读取数据,这就是一个事件。创建定时器
func NewTimer(d Duration) *Timer通过上面方法指定一个事件即可创建一个Timer,Timer一经创建便开始计时,不需要额外的启动命令。
示例:
func main() { timer := time.NewTimer(time.Second * 5) //设置超时时间5s <- timer.C fmt.Println("Time out!") }停止定时器
Timer创建后可以随时停止,停止计时器的方法如下:
func (t *Timer) Stop() bool其返回值代表定时器有没有超时:
- true:定时器超时前停止,后续不会再有事件发送。
- false:定时器超时后停止。
示例:
func main() { timer := time.NewTimer(time.Second * 5) //设置超时时间5s timer.Stop() }
重置定时器
已经过期的定时器或者已停止的定时器,可以通过重置动作重新激活,方法如下:
func (t *Timer) Reset(d Duration) bool
重置的动作本质上是先停掉定时器,再启动,其返回值也即是停掉计时器的返回值。
func main() { timer := time.NewTimer(time.Second * 5) <- timer.C fmt.Println("Time out!") timer.Stop() timer.Reset(time.Second*3) // 重置定时器 }
实现原理
每个Go应用程序都有一个协程专门负责管理所有的Timer,这个协程负责监控Timer是否过期,过期后执行一个预定义的动作,这个动作对于Timer而言就是发送当前时间到管道中。
数据结构
type Timer struct { CTimer只有两个成员:
- C:channel,上层应用根据此管道接收事件;
- r:runtimeTimer定时器,该定时器即系统管理的定时器,上层应用不可见。
runtimeTimer
任务的载体,用于监控定时任务,每创建一个Timer就创建一个runtimeTimer变量,然后把它交给系统进行监控,我们通过设置runtimeTimer过期后的行为来达到定时的目的。
源码包src/time/sleep.go:runtimeTimer定义了其数据结构:
type runtimeTimer struct { tb uintptr // 存储当前定时器的数组地址 i int // 存储当前定时器的数组下标 when int64 // 当前定时器触发时间 period int64 // 当前定时器周期触发间隔 f func(interface{}, uintptr) // 定时器触发时执行的函数 arg interface{} // 定时器触发时执行函数传递的参数一 seq uintptr // 定时器触发时执行函数传递的参数二(该参数只在网络收发场景下使用) }
创建Timer
源码实现:
func NewTimer(d Duration) *Timer { c := make(chan Time, 1) // 创建一个管道 t := &Timer{ // 构造Timer数据结构 C: c, // 新创建的管道 r: runtimeTimer{ when: when(d), // 触发时间 f: sendTime, // 触发后执行函数sendTime arg: c, // 触发后执行函数sendTime时附带的参数 }, } startTimer(&t.r) // 此处启动定时器,只是把runtimeTimer放到系统协程的堆中,由系统协程维护 return t }
NewTimer()
只是构造了一个Timer
,然后把Timer.r
通过startTimer()
交给系统协程维护。- C 是一个带1个容量的chan,这样做有什么好处呢,原因是chan 无缓冲发送数据就会阻塞,阻塞系统协程,这显然是不行的。
- 回调函数设置为
sendTime
,执行参数为channel
,sendTime
就是到点往C 里面发送当前时间的函数
sendTime实现:
//c interface{} 就是NewTimer 赋值的参数,就是channel func sendTime(c interface{}, seq uintptr) { select { case c.(chan Time)停止Timer
停止Timer,就是把Timer从系统协程中移除。函数主要实现如下:
func (t *Timer) Stop() bool { return stopTimer(&t.r) }stopTimer()即通知系统协程把该Timer移除,即不再监控。系统协程只是移除Timer并不会关闭管道,以避免用户协程读取错误。
重置Timer
重置Timer时会先把timer从系统协程中删除,修改新的时间后重新添加到系统协程中。
func (t *Timer) Reset(d Duration) bool { w := when(d) active := stopTimer(&t.r) t.r.when = w startTimer(&t.r) return active }文中关于golang的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《golang一次性定时器Timer用法及实现原理详解》文章吧,也可关注golang学习网公众号了解相关技术文章。
-
444 收藏
-
160 收藏
-
455 收藏
-
367 收藏
-
419 收藏
-
234 收藏
-
155 收藏
-
457 收藏
-
309 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习
-
- 英俊的玉米
- 这篇技术贴太及时了,大佬加油!
- 2023-01-31 19:59:08
-
- 直率的小松鼠
- 写的不错,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢up主分享文章内容!
- 2023-01-30 07:56:18
-
- 兴奋的小笼包
- 这篇文章出现的刚刚好,好细啊,受益颇多,收藏了,关注作者了!希望作者能多写Golang相关的文章。
- 2023-01-24 22:11:15
-
- 端庄的大碗
- 这篇文章内容出现的刚刚好,太细致了,感谢大佬分享,码起来,关注大佬了!希望大佬能多写Golang相关的文章。
- 2023-01-14 14:56:43
-
- 悲凉的胡萝卜
- 好细啊,mark,感谢师傅的这篇技术贴,我会继续支持!
- 2023-01-12 03:36:28