golang1.23版本之前 Timer Reset方法無(wú)法正確使用
golang1.23 之前 Reset ?到底有什么問(wèn)題
在 golang 的 time.Reset 文檔中有這么一句話,為了防止文檔更新而導(dǎo)致內(nèi)容變動(dòng),這里粘貼出來(lái):
Before Go 1.23, the only safe way to use Reset was to [Stop] and explicitly drain the timer first. See the NewTimer documentation for more details. 在Go 1.23 之前,唯一安全使用Reset函數(shù)的方式是:在使用之前調(diào)用Stop函數(shù)并且明確的從timer的channel中抽取出東西。
雖然文檔中已經(jīng)告訴了正確使用的方式,但是實(shí)際上在真正的代碼中無(wú)法達(dá)到這個(gè)要求,參考下方代碼(來(lái)源代碼來(lái)源):
//consumer go func() { // try to read from channel, block at most 5s. // if timeout, print time event and go on loop. // if read a message which is not the type we want(we want true, not false), // retry to read. timer := time.NewTimer(time.Second * 5) for { // timer may be not active, and fired if !timer.Stop() { select { case <-timer.C: //try to drain from the channel,嘗試抽取,由于使用select,因此這里可以保證:不阻塞 + 一定抽取成功 default: } } timer.Reset(time.Second * 5) //重置定時(shí)器 select { case b := <-c: if b == false { fmt.Println(time.Now(), ":recv false. continue") continue } //we want true, not false fmt.Println(time.Now(), ":recv true. return") return case <-timer.C: fmt.Println(time.Now(), ":timer expired") continue } } }()
在上面的代碼中,我們按照文檔的要求,在 timer.Reset ?之前已經(jīng)調(diào)用了 Stop ?函數(shù),且如果 Stop 成功(返回 true),還嘗試抽取 timer,看起來(lái)似乎沒(méi)問(wèn)題的代碼中仍然存在問(wèn)題。
問(wèn)題的關(guān)鍵在于:當(dāng) Ticket 觸發(fā)的時(shí)候,設(shè)置定時(shí)器狀態(tài)的操作和發(fā)送 channel 的操作并不是原子的,見 runOneTimer 函數(shù)。
異常情況:嘗試抽取 channel 在 發(fā)送 channel 之前,這樣會(huì)導(dǎo)致 Reset 之前并沒(méi)有真正的清空 timer,后一次的 timer.C 還沒(méi)到觸發(fā)時(shí)間就異常的觸發(fā)了!
golang1.23 之前到底應(yīng)該如何正確的使用 Reset?
實(shí)際上簡(jiǎn)潔點(diǎn)就這么寫,每次一個(gè)新的局部變量 Timer? 結(jié)構(gòu)體沒(méi)壓力,非要復(fù)用使用 Reset 的可讀性太差了,對(duì)維護(hù)者不友好,而且習(xí)慣了不好的寫法,哪天一不小心就寫出問(wèn)題了~
go func() { for { func() { timer := time.NewTimer(time.Second * 2) defer timer.Stop() select { case b := <-c: if !b { fmt.Println(time.Now(), "work...") } case <-timer.C: // BBB: normal receive from channel timeout event fmt.Println(time.Now(), "timeout") } }() } }()
參考:
https://tonybai.com/2016/12/21/how-to-use-timer-reset-in-golang-correctly/
到此這篇關(guān)于golang1.23版本之前 Timer Reset方法無(wú)法正確使用的文章就介紹到這了,更多相關(guān)golang Timer Reset方法無(wú)法正確使用內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- golang的time包:秒、毫秒、納秒時(shí)間戳輸出方式
- 解決golang時(shí)間字符串轉(zhuǎn)time.Time的坑
- Golang 定時(shí)器(Timer 和 Ticker),這篇文章就夠了
- golang time包下定時(shí)器的實(shí)現(xiàn)方法
- golang的時(shí)區(qū)和神奇的time.Parse的使用方法
- golang中time包之時(shí)間間隔格式化和秒、毫秒、納秒等時(shí)間戳格式輸出的方法實(shí)例
- golang 使用time包獲取時(shí)間戳與日期格式化操作
- golang time包的用法詳解
- golang time包做時(shí)間轉(zhuǎn)換操作
相關(guān)文章
go語(yǔ)言數(shù)據(jù)類型之字符串string
這篇文章介紹了go語(yǔ)言數(shù)據(jù)類型之字符串string,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-07-07golang反向代理設(shè)置host不生效的問(wèn)題解決
在使用golang的httputil做反向代理的時(shí)候,發(fā)現(xiàn)一個(gè)奇怪的現(xiàn)象,上游網(wǎng)關(guān)必須要設(shè)置host才行,不設(shè)置host的話,golang服務(wù)反向代理請(qǐng)求下游會(huì)出現(xiàn)http 503錯(cuò)誤,接下來(lái)通過(guò)本文給大家介紹golang反向代理設(shè)置host不生效問(wèn)題,感興趣的朋友一起看看吧2023-05-05Golang中json和jsoniter的區(qū)別使用示例
這篇文章主要介紹了Golang中json和jsoniter的區(qū)別使用示例,本文給大家分享兩種區(qū)別,結(jié)合示例代碼給大家介紹的非常詳細(xì),感興趣的朋友跟隨小編一起看看吧2023-12-12GO語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單TCP服務(wù)的方法
這篇文章主要介紹了GO語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單TCP服務(wù)的方法,實(shí)例分析了Go語(yǔ)言實(shí)現(xiàn)TCP服務(wù)的技巧,需要的朋友可以參考下2015-03-03golang實(shí)現(xiàn)簡(jiǎn)易的分布式系統(tǒng)方法
這篇文章主要介紹了golang實(shí)現(xiàn)簡(jiǎn)易的分布式系統(tǒng)方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-10-10Golang WebView跨平臺(tái)的桌面應(yīng)用庫(kù)的使用
Golang WebView是一個(gè)強(qiáng)大的桌面應(yīng)用庫(kù),本文介紹了Golang WebView的特點(diǎn)和使用方法,并列舉示例詳細(xì)的介紹了其在實(shí)際項(xiàng)目中的應(yīng)用,具有一定的參考價(jià)值,感興趣的可以了解一下2024-03-03淺析Golang切片截取功能與C++的vector區(qū)別
這篇文章主要介紹了Golang中切片的截取功能與C++中的vector有什么區(qū)別,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)吧2022-10-10go實(shí)現(xiàn)圖片拼接與文字書寫的方法實(shí)例
這篇文章主要給大家介紹了關(guān)于go實(shí)現(xiàn)圖片拼接與文字書寫的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2022-01-01