浅析go中Ticker,Timer和Tick的用法与区别
作者:nil
写在前面
在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个函数就是NewTicker
和Stop
。
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内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!