網站首頁 編程語言 正文
前言
定時器在Go語言應用中使用非常廣泛,Go語言的標準庫里提供兩種類型的計時器,一種是一次性的定時器Timer
,另外一種是周期性的定時器Ticker
。本文主要來看一下Timer
的用法和實現原理,需要的朋友可以參考以下內容,希望對大家有幫助。
Timer
Timer
是一種單一事件的定時器,即經過指定的時間后觸發一個事件,因為Timer
只執行一次就結束,所以稱為單一事件,這個事件通過其本身提供的channel
進行通知觸發。
timer結構體
通過src/time.sleep.go:Timer
定義了Timer
數據結構:
// Timer代表一次定時,時間到達后僅執行一個事件。 type Timer struct { C <-chan Time r runtimeTimer }
它提供了一個channel
,在定時時間到達之前,沒有數據寫入timer.C
會一直阻塞,直到時間到達,向channel
寫入系統時間,阻塞解除,可以從中讀取數據,這就是一個事件。
創建定時器
func NewTimer(d Duration) *Timer
通過上面方法指定一個事件即可創建一個Timer,Timer一經創建便開始計時,不需要額外的啟動命令。
示例:
func main() { timer := time.NewTimer(time.Second * 5) //設置超時時間5s <- timer.C fmt.Println("Time out!") }
停止定時器
Timer創建后可以隨時停止,停止計時器的方法如下:
func (t *Timer) Stop() bool
其返回值代表定時器有沒有超時:
- true:定時器超時前停止,后續不會再有事件發送。
- false:定時器超時后停止。
示例:
func main() { timer := time.NewTimer(time.Second * 5) //設置超時時間5s timer.Stop() }
重置定時器
已經過期的定時器或者已停止的定時器,可以通過重置動作重新激活,方法如下:
func (t *Timer) Reset(d Duration) bool
重置的動作本質上是先停掉定時器,再啟動,其返回值也即是停掉計時器的返回值。
func main() { timer := time.NewTimer(time.Second * 5) <- timer.C fmt.Println("Time out!") timer.Stop() timer.Reset(time.Second*3) // 重置定時器 }
實現原理
每個Go應用程序都有一個協程專門負責管理所有的Timer,這個協程負責監控Timer是否過期,過期后執行一個預定義的動作,這個動作對于Timer而言就是發送當前時間到管道中。
數據結構
type Timer struct { C <-chan Time r runtimeTimer }
Timer只有兩個成員:
- C:channel,上層應用根據此管道接收事件;
- r:runtimeTimer定時器,該定時器即系統管理的定時器,上層應用不可見。
runtimeTimer
任務的載體,用于監控定時任務,每創建一個Timer就創建一個runtimeTimer變量,然后把它交給系統進行監控,我們通過設置runtimeTimer過期后的行為來達到定時的目的。
源碼包src/time/sleep.go:runtimeTimer定義了其數據結構:
type runtimeTimer struct { tb uintptr // 存儲當前定時器的數組地址 i int // 存儲當前定時器的數組下標 when int64 // 當前定時器觸發時間 period int64 // 當前定時器周期觸發間隔 f func(interface{}, uintptr) // 定時器觸發時執行的函數 arg interface{} // 定時器觸發時執行函數傳遞的參數一 seq uintptr // 定時器觸發時執行函數傳遞的參數二(該參數只在網絡收發場景下使用) }
創建Timer
源碼實現:
func NewTimer(d Duration) *Timer { c := make(chan Time, 1) // 創建一個管道 t := &Timer{ // 構造Timer數據結構 C: c, // 新創建的管道 r: runtimeTimer{ when: when(d), // 觸發時間 f: sendTime, // 觸發后執行函數sendTime arg: c, // 觸發后執行函數sendTime時附帶的參數 }, } startTimer(&t.r) // 此處啟動定時器,只是把runtimeTimer放到系統協程的堆中,由系統協程維護 return t }
-
NewTimer()
只是構造了一個Timer
,然后把Timer.r
通過startTimer()
交給系統協程維護。 - C 是一個帶1個容量的chan,這樣做有什么好處呢,原因是chan 無緩沖發送數據就會阻塞,阻塞系統協程,這顯然是不行的。
- 回調函數設置為
sendTime
,執行參數為channel
,sendTime
就是到點往C 里面發送當前時間的函數
sendTime實現:
//c interface{} 就是NewTimer 賦值的參數,就是channel func sendTime(c interface{}, seq uintptr) { select { case c.(chan Time) <- Now(): //寫不進去的話,C 已滿,走default 分支 default: } }
停止Timer
停止Timer,就是把Timer從系統協程中移除。函數主要實現如下:
func (t *Timer) Stop() bool { return stopTimer(&t.r) }
stopTimer()即通知系統協程把該Timer移除,即不再監控。系統協程只是移除Timer并不會關閉管道,以避免用戶協程讀取錯誤。
重置Timer
重置Timer時會先把timer從系統協程中刪除,修改新的時間后重新添加到系統協程中。
func (t *Timer) Reset(d Duration) bool { w := when(d) active := stopTimer(&t.r) t.r.when = w startTimer(&t.r) return active }
原文鏈接:https://juejin.cn/post/7134606069203992583
相關推薦
- 2022-04-23 Python繪制燈籠的示例代碼_python
- 2022-09-16 Pandas?類型轉換astype()的實現_python
- 2022-07-31 python虛擬機解釋器及運行過程_python
- 2022-10-27 詳解Python中enumerate函數的使用_python
- 2023-06-13 C語言中函數返回值不一致問題_C 語言
- 2022-09-03 詳解.NET主流的幾款重量級?ORM框架_實用技巧
- 2022-07-04 Python中字典的緩存池_python
- 2022-02-25 Oracle工具PL/SQL的基本語法_oracle
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支