Go 語言一次性定時器使用方式和實現(xiàn)原理
?1.介紹
在 Go 語言標(biāo)準(zhǔn)庫 time? 包中的 Timer 類型,它是表示單一事件的計時器,也就是說它是一次性定時器。
在 Go 語言項目開發(fā)中,定時器使用廣泛,本文我們介紹 Go 語言中怎么使用 Timer,以及它的實現(xiàn)原理。
2.使用方式
使用 Timer? 一次性定時器,需要導(dǎo)入 time 包,創(chuàng)建定時器的方式有兩種,分別是:
func NewTimer(d Duration) *Timer
使用 func NewTimer? 創(chuàng)建 Timer?,入?yún)⑹嵌〞r器的等待時間,時間到達時,發(fā)送當(dāng)前時間到 channel中。
示例代碼:
輸出結(jié)果:
通過閱讀上面這段代碼,我們可以發(fā)現(xiàn)我們定義了一個 2s? 后執(zhí)行的定時器 timer?,然后使用 select? 讀取 timer.C 中的數(shù)據(jù),當(dāng)讀取到數(shù)據(jù)時,執(zhí)行特定業(yè)務(wù)邏輯代碼。
func AfterFunc(d Duration, f func()) *Timer
使用 func AfterFunc? 創(chuàng)建 Timer,入?yún)⑹嵌〞r器等待時間,和時間到達時執(zhí)行的函數(shù)。
示例代碼:
閱讀上面這段代碼,細(xì)心的讀者朋友們可能已經(jīng)發(fā)現(xiàn),我們在代碼末尾使用 time.Sleep()?,這是因為 time.AfterFunc() 是異步執(zhí)行的,所以需要等待協(xié)成退出。
3.實現(xiàn)原理
我們在源碼中查看 Timer? 的數(shù)據(jù)結(jié)構(gòu),發(fā)現(xiàn)它包含兩個字段,其中一個是可導(dǎo)出字段 C?,這是一個 Time?類型的 chan?;另一個是不可導(dǎo)出字段 r?,這是一個 runtimeTimer 類型。
實際上,每個 Go 應(yīng)用程序底層都會有一個特定的協(xié)程管理 Timer?,該協(xié)程(底層協(xié)程)監(jiān)控到某個 Timer? 指定的時間到達時,就會將當(dāng)前時間發(fā)送到 C? 中,然后上層讀取到 C 中的數(shù)據(jù)時,執(zhí)行相關(guān)業(yè)務(wù)邏輯代碼。
底層協(xié)程監(jiān)控 Timer? 的 r? 字段中的數(shù)據(jù),我們在源碼中查看一下 runtimeTimer 的數(shù)據(jù)結(jié)構(gòu):
閱讀上面這段代碼,我們可以發(fā)現(xiàn) runtimeTimer? 中包含的所有字段,我們重點了解 when、f? 和 arg。
- when:定時器執(zhí)行時間。
- f:定時器執(zhí)行的回調(diào)函數(shù)。
- arg:定時器執(zhí)行的回調(diào)函數(shù)的參數(shù)。
在簡單了解 Timer? 的數(shù)據(jù)結(jié)構(gòu)之后,我們在源碼中查看一下 func NewTimer 的代碼:
閱讀上面這段代碼,我們可以發(fā)現(xiàn) func NewTimer? 的實現(xiàn)非常簡單,它實際上就是構(gòu)造了一個 Timer?,然后把 Timer.r? 傳參給 startTimer()?,除了 startTimer()? 函數(shù)外,還有兩個函數(shù),分別是 when()?和 sendTime?,其中 when()? 是計算計時器的執(zhí)行時間,sendTime 是計時器時間到達時執(zhí)行的事件(實際上就是將當(dāng)前時間寫入通道中)。
sendTime 源碼:
我們已經(jīng)了解到,func NewTimer? 將構(gòu)造的 Timer.r? 傳參給 startTimer()?,它負(fù)責(zé)把 runtimeTimer?寫入底層協(xié)程的數(shù)組中(如果底層協(xié)程未運行,它將會啟動底層協(xié)程),將 Timer? 交給底層協(xié)程監(jiān)控,也就是上面講到的,當(dāng)?shù)讓訁f(xié)程監(jiān)控到某個 Timer 指定時間到達時,將當(dāng)前時間發(fā)送到它的通道中。
4.總結(jié)
本文我們介紹 Go 語言標(biāo)準(zhǔn)庫 time? 包提供的一次性定時器 Timer,不僅介紹了它的使用方式,還介紹了它的實現(xiàn)原理。