golang定时器Timer的用法和实现原理是什么

本篇内容介绍了“golang定时器Timer的用法和实现原理是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

    Timer

    Timer是一种单一事件的定时器,即经过指定的时间后触发一个事件,因为Timer只执行一次就结束,所以称为单一事件,这个事件通过其本身提供的channel进行通知触发。

    timer结构体

    通过src/time.sleep.go:Timer定义了Timer数据结构:

    // Timer代表一次定时,时间到达后仅执行一个事件。
    type Timer struct {
        C <-chan Time
        r runtimeTimer
    }

    它提供了一个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 {
        C <-chan Time
        r runtimeTimer
    }

    Timer只有两个成员:

    • 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,执行参数为channelsendTime就是到点往C 里面发送当前时间的函数

    sendTime实现:

    //c interface{} 就是NewTimer 赋值的参数,就是channel
    func sendTime(c interface{}, seq uintptr) {
        select {
        case c.(chan Time) <- Now(): //写不进去的话,C 已满,走default 分支
        default:
        }
    }
    停止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定时器Ticker

    time包下有一个Ticker结构体

    // Ticker保管一个通道,并每隔一段时间向其传递"tick"。
    type Ticker struct {
    	C <-chan Time // 周期性传递时间信息的通道.
    	r runtimeTimer
    }
    func NewTicker(d Duration) *Ticker{}

    NewTicker返回一个新的Ticker,该Ticker包含一个通道字段,并会每隔时间段d,就向该通道发送当时的时间。它会调整时间间隔或者丢弃tick信息以适应反应慢的接收者。如果d<=0会panic。关闭该Ticker可以释放相关资源。

    func (t *Ticker) Stop()

    Stop关闭一个Ticker。在关闭后,将不会发送更多的tick信息。Stop不会关闭通道t.C,以避免从该通道的读取不正确的成功。

    例子

    package main
    
    import (
    	"fmt"
    	"time"
    )
    
    func main() {
    	t := time.NewTicker(5 * time.Second) //创建定时器
    	defer t.Stop()
        
    	go sync(t)
    	select {
    
    	}
    }
    
    func sync(t *time.Ticker) {
    	for {
    		// 每5秒中从chan t.C 中读取一次
    		<-t.C
    		fmt.Println("执行数据备份任务:", time.Now().Format("2006-01-02 15:04:05"))
    	}
    }

    “golang定时器Timer的用法和实现原理是什么”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注编程之家网站,小编将为大家输出更多高质量的实用文章!

    版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

    相关推荐


    类型转换 1、int转string 2、string转int 3、string转float 4、用户结构类型转换
    package main import s &quot;strings&quot; import &quot;fmt&quot; var p = fmt.Println func main() { p(&quot;Contains: &quot;, s.Contains(&quot;test&quo
    类使用:实现一个people中有一个sayhi的方法调用功能,代码如下: 接口使用:实现上面功能,代码如下:
    html代码: beego代码:
    1、读取文件信息: 2、读取文件夹下的所有文件: 3、写入文件信息 4、删除文件,成功返回true,失败返回false
    配置环境:Windows7+推荐IDE:LiteIDEGO下载地址:http://www.golangtc.com/downloadBeego开发文档地址:http://beego.me/docs/intro/ 安装步骤: 一、GO环境安装 二、配置系统变量 三、Beego安装 一、GO环境安装 根
    golang获取程序运行路径:
    Golang的文档和社区资源:为什么它可以帮助开发人员快速上手?
    Golang:AI 开发者的实用工具
    Golang的标准库:为什么它可以大幅度提高开发效率?
    Golang的部署和运维:如何将应用程序部署到生产环境中?
    高性能AI开发:Golang的优势所在
    本篇文章和大家了解一下go语言开发优雅得关闭协程的方法。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。1.简介本文将介绍首先为什么需要主...
    这篇文章主要介绍了Go关闭goroutine协程的方法,具有一定借鉴价值,需要的朋友可以参考下。下面就和我一起来看看吧。1.简介本文将介绍首先为什么需要主动关闭gor...
    本篇文章和大家了解一下go关闭GracefulShutdown服务的几种方法。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。目录Shutdown方法Regi...
    这篇文章主要介绍了Go语言如何实现LRU算法的核心思想和实现过程,具有一定借鉴价值,需要的朋友可以参考下。下面就和我一起来看看吧。GO实现Redis的LRU例子常
    今天小编给大家分享的是Go简单实现多租户数据库隔离的方法,相信很多人都不太了解,为了让大家更加了解,所以给大家总结了以下内容,一起往下看吧。一定会...
    这篇“Linux系统中怎么安装NSQ的Go语言客户端”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希
    本文小编为大家详细介绍“怎么在Go语言中实现锁机制”,内容详细,步骤清晰,细节处理妥当,希望这篇“怎么在Go语言中实现锁机制”文章能帮助大家解决疑惑,下面...
    今天小编给大家分享一下Go语言中interface类型怎么使用的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考