浅析go中Ticker,Timer和Tick的用法与区别

 更新时间:2023年10月22日 08:04:22   作者:nil  
在go面试的时候,面试官经常会问time包的Ticker,Timer以及Tick的区别,一般在超时控制的时候用的比较多,今天就跟随小编一起来详细学一下这几个的区别吧

写在前面

在go面试的时候,面试官经常会问time包的Ticker,Timer以及Tick的区别,一般在超时控制的时候用的比较多。今天就来详细学一下这几个的区别

Ticker

Ticker是周期性定时器,即周期性的触发一个事件,它会以一个间隔(interval)往channel发送一个事件(当前时间),而channel的接收者可以以固定的时间间隔从channel中读取事件。通过Ticker本身提供的管道将事件传递出去

示例

func main() {
    t1 := time.NewTicker(3 * time.Second)
    defer t1.Stop()
    for {
       select {
       case <-t1.C:
          fmt.Printf("%s\n", time.Now())
       }
    }
}

结果

2023-10-21 18:53:30.629901 +0800 CST m=+3.001061072
2023-10-21 18:53:33.629444 +0800 CST m=+6.000526969
2023-10-21 18:53:36.629429 +0800 CST m=+9.000436237

通过NewTicker创建一个Ticker,然后通过通过for+select实现定时执行。记住Ticker一定要调用Stop方法,关闭chan,否则容易造成内存泄露。Ticker最常用的2个函数就是NewTickerStop

Ticker主要用于定期循环执行某一个操作。

Timer

用于执行一次性任务。在指定时间间隔之后再下chan发送一个事件(当前时间)

func main() {
    t1 := time.NewTimer(3 * time.Second)
    defer t1.Stop()
    for {
       t1.Reset(3 * time.Second)
       select {
       case <-t1.C:
          fmt.Printf("%s\n", time.Now())
       }
    }
}

用Timer也能实现Ticker循环执行的功能,但是每次必须调用Reset方法,否则会死锁,因为Timer只会执行一次。不过Timer一般不用于循环执行,它用于超时控制,比如调用某个rpc,如果超过3s则终止调用。

示例

func main() {
    t1 := time.NewTimer(3 * time.Second)
    defer t1.Stop()

    ctx, cancel := mockRPC()
    select {
    case <-ctx.Done():
       fmt.Printf("rpc end\n")
       break
    case <-t1.C:
       // t1时间到了,rpc还没执行完,则执行cancel()并退出
       fmt.Printf("%s\n", time.Now())
       cancel()
    }
}

// mockRPC. 异步执行5s钟,如果已经结束了则停止执行
func mockRPC() (context.Context, context.CancelFunc) {
    ctx, cancel := context.WithCancel(context.Background())
    go func() {
       for i := 0; i < 5; i++ {
          select {
          case <-ctx.Done():
             break
          default:
             fmt.Printf("%d\n", i)
             time.Sleep(time.Second)
          }
       }
    }()

    return ctx, cancel
}

结果

0
1
2
2023-10-21 19:18:53.565153 +0800 CST m=+3.000558307

Tick

Tick其实最简单,直接看它的源代码

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

发现Tick其实就是一个Ticker的C,所以使用起来更方便

func main() {
    t1 := time.Tick(3 * time.Second)
    for {
       select {
       case <-t1:
          fmt.Printf("%s\n", time.Now())
       }
    }
}

但是有一个问题,由于Tick()函数没有返回Ticker,我们无法关闭它,所以这个t1对应的Ticker用于无法关闭,会造成内存泄露。一般不推荐用Tick,直接使用Ticker就好了。应该只能在程序结束的时候会回收内存,但是Tick和Ticker区别这么小,不知道为什么要定义Tick()这个函数。

总结

  • Ticker是一个定时器。
  • Timer是一个延迟执行器,也可以通过Reset()来实现定时器。
  • Tick()是一个函数,返回Ticker的chan,不推荐使用。

到此这篇关于浅析go中Ticker,Timer和Tick的用法与区别的文章就介绍到这了,更多相关go Ticker Timer Tick内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Go标准库日志打印及同时输出到控制台与文件

    Go标准库日志打印及同时输出到控制台与文件

    Go语言内置的log包实现了简单的日志服务,下面这篇文章主要给大家介绍了关于Go标准库日志打印及同时输出到控制台与文件的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-11-11
  • Go web入门Go pongo2模板引擎

    Go web入门Go pongo2模板引擎

    这篇文章主要为大家介绍了Go web编程入门Go pongo2模板引擎使用详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-05-05
  • Go中跨域Cors中间件的实现

    Go中跨域Cors中间件的实现

    本文主要介绍了Go中跨域Cors中间件的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-06-06
  • golang协程关闭踩坑实战记录

    golang协程关闭踩坑实战记录

    协程(coroutine)是Go语言中的轻量级线程实现,下面这篇文章主要给大家介绍了关于golang协程关闭踩坑的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-03-03
  • Go映射的使用

    Go映射的使用

    Go提供了另一个重要的数据类型,称为map,它将唯一键映射到值,本文主要介绍了Go映射的使用,包括声明映射、初始化映射、操作映射等,感兴趣的可以了解一下
    2023-11-11
  • GO开发编辑器安装图文详解

    GO开发编辑器安装图文详解

    这篇文章主要介绍了GO开发编辑器安装,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-01-01
  • Go Sentinel 动态数据源配置指南(示例详解)

    Go Sentinel 动态数据源配置指南(示例详解)

    本文介绍了如何使用Go语言配置Sentinel的动态数据源,并通过本地文件和Nacos两种方式实现动态配置,通过这种方式,可以灵活地管理和更新限流规则,提升系统的稳定性和响应速度,感兴趣的朋友跟随小编一起看看吧
    2025-01-01
  • Systemd集成Golang二进制程序的方法

    Systemd集成Golang二进制程序的方法

    这篇文章主要介绍了Systemd集成Golang二进制程序的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-10-10
  • Go语言{}大括号的特殊用法实例探究

    Go语言{}大括号的特殊用法实例探究

    这篇文章主要为大家介绍了Go语言{}大括号的特殊用法实例探究,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2024-01-01
  • golang并发执行的几种方式小结

    golang并发执行的几种方式小结

    本文主要介绍了golang并发执行的几种方式小结,主要包括了Channel,WaitGroup ,Context,使用这三种机制中的一种或者多种可以达到并发控制很好的效果,具有一定的参考价值,感兴趣的可以了解一下
    2023-08-08

最新评论