Golang 定時器的終止與重置實現(xiàn)
昨日有讀者對定時器的終止有疑問,今天我們來聊一聊定時器的終止與重置吧!
定時器是一種通過設(shè)置一項任務(wù),在未來的某個時刻執(zhí)行該任務(wù)的機(jī)制。
定時器的種類通常只有兩種,一種是只執(zhí)行一次的延時模式,一種是每隔一段時間執(zhí)行一次的間隔模式。
在現(xiàn)代編程語言中,定時器幾乎是標(biāo)配。除了設(shè)置定時器外,還需要有提供定時器的方法。
比如在 JavaScript 中,提供了 setTimeout、setInterval、clearTimeout 和 clearInterval 四個 API,相比較而言是比較簡單的。Go 語言中定時器的 API 就比較完善,所有的 API 都在 time 包中。
先看下面一段代碼:
func main() { timer := time.NewTimer(3 * time.Second) fmt.Println(time.Now(),"炸彈將于3秒后引爆") timer.Stop() fmt.Println("定時炸彈已拆除,定時器失效") t := <-timer.C fmt.Println("炸彈引爆于",t) }
先來看看運(yùn)行結(jié)果
2021-08-25 10:08:34.706412 +0800 CST m=+0.023017601 炸彈將于3秒后引爆
定時炸彈已拆除,定時器失效
fatal error: all goroutines are asleep - deadlock!
我們可以趁定時器時間未到而使用Stop來將定時器終止,如果定時器已被叫停,其時間管道永遠(yuǎn)讀不出數(shù)據(jù)了,如果強(qiáng)制讀取,就會出現(xiàn)死鎖。因為使用Stop就是停止往管道里面寫數(shù)據(jù)了,或者可以這樣說,就是管道里面的數(shù)據(jù)已經(jīng)讀完了,使用time.NewTimer(3 * time.Second)就是往管道里面寫數(shù)據(jù)。
我們在來看一個有趣的例子。
func main() { timer := time.NewTimer(1 * time.Second) fmt.Println(time.Now()) time.Sleep(2 * time.Second) fmt.Println(time.Now()) timer.Reset(10*time.Second) fmt.Println("炸彈引爆于",<-timer.C) }
現(xiàn)在,思考一下,炸彈是什么時候引爆的!
想知道答案嗎?不要著急,不要著急,休息,休息一會兒,答案馬上揭曉
我們來看看運(yùn)行結(jié)果吧:
2021-08-25 10:15:16.8406335 +0800 CST m=+0.014999801
2021-08-25 10:15:18.906213 +0800 CST m=+2.080579301
炸彈引爆于 2021-08-25 10:15:17.8522233 +0800 CST m=+1.026589601
是不是和你想的一樣?如果不是,沒關(guān)系,聽我細(xì)細(xì)道來。
因為time.sleep()是讓主協(xié)程睡大覺,而timer.C讀的那條管道的協(xié)程是獨立的。所以你讓主協(xié)程睡大覺并不會影響定時器的計時,就相當(dāng)于一個定時炸彈要引爆了,你馬上把手表的時間往后調(diào),但是定時炸彈上的數(shù)字時間不會因為手表上的時間往后調(diào)而往后調(diào)。
誒!這時你會說我不是重置了嗎?
但是定時器超時了,那么重置就不起作用了,你想一想,定時炸彈都爆炸了,你去重置還有效嗎?
如果我們將定時器的時間調(diào)到3秒,就是這樣:
timer := time.NewTimer(3 * time.Second)
那么輸出結(jié)果會怎樣?
2021-08-25 10:26:21.1299417 +0800 CST m=+0.020983301
2021-08-25 10:26:23.2191128 +0800 CST m=+2.110154401
炸彈引爆于 2021-08-25 10:26:33.227692 +0800 CST m=+12.118733601
設(shè)置定時器后2秒,主協(xié)程才執(zhí)行到Reset(),所以炸彈是在設(shè)置定時器12秒后才爆炸的。
有趣的是,當(dāng)我查看Reset()的源碼時,發(fā)現(xiàn)了這樣一段注釋:
// Reset should be invoked only on stopped or expired timers with drained channels. // If a program has already received a value from t.C, the timer is known // to have expired and the channel drained, so t.Reset can be used directly. // If a program has not yet received a value from t.C, however, // the timer must be stopped and—if Stop reports that the timer expired // before being stopped—the channel explicitly drained: // // if !t.Stop() { // <-t.C // } // t.Reset(d)
根據(jù)我的理解,大意是這樣的,如果計時器已經(jīng)過期,并且t.C已經(jīng)被讀完了,那么可以直接使用Reset。而如果程序Reset之前未從t.C中讀取過值的話,就需要調(diào)用Stop來結(jié)束定時器,才能使用reset。
到此這篇關(guān)于Golang 定時器的終止與重置實現(xiàn)的文章就介紹到這了,更多相關(guān)Golang 定時器終止與重置內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go語言同步等待組sync.WaitGroup結(jié)構(gòu)體對象方法詳解
這篇文章主要為大家介紹了Go語言同步等待組sync.WaitGroup結(jié)構(gòu)體對象方法詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08使用Go Validator有效驗證數(shù)據(jù)示例分析
作為一名開發(fā)者,確保Go應(yīng)用中處理的數(shù)據(jù)是有效和準(zhǔn)確的非常重要,Go Validator是一個開源的數(shù)據(jù)驗證庫,為Go結(jié)構(gòu)體提供強(qiáng)大且易于使用的數(shù)據(jù)驗證功能,本篇文章將介紹Go Validator庫的主要特點以及如何在Go應(yīng)用中使用它來有效驗證數(shù)據(jù)2023-12-12輕松入門:使用Golang開發(fā)跨平臺GUI應(yīng)用
Golang是一種強(qiáng)大的編程語言,它的并發(fā)性和高性能使其成為開發(fā)GUI桌面應(yīng)用的理想選擇,Golang提供了豐富的標(biāo)準(zhǔn)庫和第三方庫,可以輕松地創(chuàng)建跨平臺的GUI應(yīng)用程序,通過使用Golang的GUI庫,開發(fā)人員可以快速構(gòu)建具有豐富用戶界面和交互功能的應(yīng)用程序,需要的朋友可以參考下2023-10-10Go語言crypto包創(chuàng)建自己的密碼加密工具實現(xiàn)示例
Go語言借助它的簡單性和強(qiáng)大的標(biāo)準(zhǔn)庫,實現(xiàn)一個自己的密碼加密工具,本文將會結(jié)合代碼示例深入探討如何使用Go語言的crypto包來實現(xiàn)自己的加密工具2023-11-11利用Golang解析json數(shù)據(jù)的方法示例
Go提供了原生的JSON庫,并且與語言本身有效的集成在了一起。下面這篇文章將給大家介紹關(guān)于利用Golang解析json數(shù)據(jù)的方法,文中給出了詳細(xì)的示例代碼供大家參考學(xué)習(xí),需要的朋友們下面跟著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2017-07-07go語言使用pipe讀取子進(jìn)程標(biāo)準(zhǔn)輸出的方法
這篇文章主要介紹了go語言使用pipe讀取子進(jìn)程標(biāo)準(zhǔn)輸出的方法,實例分析了Go語言針對進(jìn)程操作的技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-03-03