请选择 进入手机版 | 继续访问电脑版

[Golang] Golang 定时器的停止 与重置实现

[复制链接]
查看97 | 回复14 | 2021-9-15 02:19:59 | 显示全部楼层 |阅读模式

昨日有读者对定时器的克制 有疑问,本日 我们来聊一聊定时器的克制 与重置吧!

定时器是一种通过设置一项使命 ,在将来 的某个时候 实行 该使命 的机制。

定时器的种类通常只有两种,一种是只实行 一次的延时模式,一种是每隔一段时间实行 一次的间隔模式。

在当代 编程语言中,定时器几乎是标配。除了设置定时器外,还必要 有提供定时器的方法。

比如在 JavaScript 中,提供了 setTimeout、setInterval、clearTimeout 和 clearInterval 四个 API,相比较而言是比较简单的。Go 语言中定时器的 API 就比较美满 ,全部 的 API 都在 time 包中。

先看下面一段代码:

  1. func main() {
  2. timer := time.NewTimer(3 * time.Second)
  3. fmt.Println(time.Now(),"炸弹将于3秒后引爆")
  4. timer.Stop()
  5. fmt.Println("定时炸弹已拆除,定时器失效")
  6. t := <-timer.C
  7. fmt.Println("炸弹引爆于",t)
  8. }
复制代码

先来看看运行效果

  1. 2021-08-25 10:08:34.706412 +0800 CST m=+0.023017601 炸弹将于3秒后引爆
  2. 定时炸弹已拆除,定时器失效
  3. fatal error: all goroutines are asleep - deadlock!
复制代码

我们可以趁定时器时间未到而使用 Stop来将定时器克制 ,假如 定时器已被叫停,当时 间管道永久 读不出数据了,假如 逼迫 读取,就会出现死锁。由于 使用 Stop就是克制 往管道内里 写数据了,或者可以如许 说,就是管道内里 的数据已经读完了,使用 time.NewTimer(3 * time.Second)就是往管道内里 写数据。

我们在来看一个风趣 的例子。

  1. func main() {
  2. timer := time.NewTimer(1 * time.Second)
  3. fmt.Println(time.Now())
  4. time.Sleep(2 * time.Second)
  5. fmt.Println(time.Now())
  6. timer.Reset(10*time.Second)
  7. fmt.Println("炸弹引爆于",<-timer.C)
  8. }
复制代码

如今 ,思考 一下,炸弹是什么时间 引爆的!

想知道答案吗?不要发急 ,不要发急 ,苏息 ,苏息 一会儿,答案立刻 发表

我们来看看运行效果 吧:

  1. 2021-08-25 10:15:16.8406335 +0800 CST m=+0.014999801
  2. 2021-08-25 10:15:18.906213 +0800 CST m=+2.080579301
  3. 炸弹引爆于 2021-08-25 10:15:17.8522233 +0800 CST m=+1.026589601
复制代码

是不是和你想的一样?假如 不是,不要紧 ,听我细细道来。
由于 time.sleep()是让主协程睡大觉,而timer.C读的那条管道的协程是独立的。以是 你让主协程睡大觉并不会影响定时器的计时,就相称 于一个定时炸弹要引爆了,你立刻 把手表的时间以后 调,但是定时炸弹上的数字时间不会由于 手表上的时间以后 调而以后 调。

诶!这时你会说我不是重置了吗?
但是定时器超时了,那么重置就不起作用了,你想一想,定时炸弹都爆炸了,你去重置还有用 吗?
假如 我们将定时器的时间调到3秒,就是如许 :

  1. timer := time.NewTimer(3 * time.Second)
复制代码

那么输出效果 会怎样?

  1. 2021-08-25 10:26:21.1299417 +0800 CST m=+0.020983301
  2. 2021-08-25 10:26:23.2191128 +0800 CST m=+2.110154401
  3. 炸弹引爆于 2021-08-25 10:26:33.227692 +0800 CST m=+12.118733601
复制代码

设置定时器后2秒,主协程才实行 到Reset(),以是 炸弹是在设置定时器12秒后才爆炸的。
风趣 的是,当我查看Reset()的源码时,发现了如许 一段解释 :

  1. // Reset should be invoked only on stopped or expired timers with drained channels.
  2. // If a program has already received a value from t.C, the timer is known
  3. // to have expired and the channel drained, so t.Reset can be used directly.
  4. // If a program has not yet received a value from t.C, however,
  5. // the timer must be stopped and—if Stop reports that the timer expired
  6. // before being stopped—the channel explicitly drained:
  7. //
  8. // if !t.Stop() {
  9. // <-t.C
  10. // }
  11. // t.Reset(d)
复制代码

根据我的明确 ,大意是如许 的,假如 计时器已颠末 期,并且t.C已经被读完了,那么可以直接使用 Reset。而假如 程序Reset之前未从t.C中读取过值的话,就必要 调用Stop来竣事 定时器,才能使用 reset。

到此这篇关于Golang 定时器的克制 与重置实现的文章就先容 到这了,更多干系 Golang 定时器克制 与重置内容请搜刮 脚本之家从前 的文章或继续欣赏 下面的干系 文章盼望 大家以后多多支持脚本之家!


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
回复

使用道具 举报

avatar 饺子姑娘 | 2021-9-25 07:08:48 | 显示全部楼层
admin楼主的文笔不错!
回复

使用道具 举报

avatar 又见阿凡提彻 | 2021-10-8 15:51:08 | 显示全部楼层
有机会找admin楼主好好聊聊!
回复

使用道具 举报

avatar jj13904198394 | 2021-10-9 10:41:02 | 显示全部楼层
楼上的这是啥态度呢?
回复

使用道具 举报

avatar ggp27 | 2021-10-13 08:33:02 | 显示全部楼层
支持一下,下面的保持队形!
回复

使用道具 举报

avatar 江左岸右郧 | 2021-10-13 08:33:05 | 显示全部楼层
我对admin楼主的敬仰犹如滔滔江水绵延不绝!
回复

使用道具 举报

avatar poney | 7 天前 | 显示全部楼层
今天上网不回帖,回帖就回精华帖!
回复

使用道具 举报

好多兽医在广场上义诊,admin楼主去看看吧!
回复

使用道具 举报

记得吃药!
回复

使用道具 举报

看帖不回帖的人就是耍流氓,我回复了!
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则