成人性生交大片免费看视频r_亚洲综合极品香蕉久久网_在线视频免费观看一区_亚洲精品亚洲人成人网在线播放_国产精品毛片av_久久久久国产精品www_亚洲国产一区二区三区在线播_日韩一区二区三区四区区区_亚洲精品国产无套在线观_国产免费www

主頁 > 知識庫 > golang 定時任務(wù)方面time.Sleep和time.Tick的優(yōu)劣對比分析

golang 定時任務(wù)方面time.Sleep和time.Tick的優(yōu)劣對比分析

熱門標(biāo)簽:評價高的400電話辦理 壽光微信地圖標(biāo)注 涿州代理外呼系統(tǒng) 外呼系統(tǒng)顯本地手機(jī)號 外呼系統(tǒng)用什么卡 阿克蘇地圖標(biāo)注 電話機(jī)器人軟件免費(fèi) 百度地圖標(biāo)注后傳給手機(jī) excel地圖標(biāo)注分布數(shù)據(jù)

golang 寫循環(huán)執(zhí)行的定時任務(wù),常見的有以下三種實(shí)現(xiàn)方式

1、time.Sleep方法:

for {
   time.Sleep(time.Second)
   fmt.Println("我在定時執(zhí)行任務(wù)")
}

2、time.Tick函數(shù):

t1:=time.Tick(3*time.Second)
for {
   select {
   case -t1:
      fmt.Println("t1定時器")
   }
}

3、其中Tick定時任務(wù)

也可以先使用time.Ticker函數(shù)獲取Ticker結(jié)構(gòu)體,然后進(jìn)行阻塞監(jiān)聽信息,這種方式可以手動選擇停止定時任務(wù),在停止任務(wù)時,減少對內(nèi)存的浪費(fèi)。

t:=time.NewTicker(time.Second)
for {
   select {
   case -t.C:
      fmt.Println("t1定時器")
      t.Stop()
   }
}

其中第二種和第三種可以歸為同一類

這三種定時器的實(shí)現(xiàn)原理

一般來說,你在使用執(zhí)行定時任務(wù)的時候,一般旁人會勸你不要使用time.Sleep完成定時任務(wù),但是為什么不能使用Sleep函數(shù)完成定時任務(wù)呢,它和Tick函數(shù)比,有什么劣勢呢?這就需要我們?nèi)ヌ接戦喿x一下源碼,分析一下它們之間的優(yōu)劣性。

首先,我們研究一下Tick函數(shù),func Tick(d Duration) -chan Time

調(diào)用Tick函數(shù)會返回一個時間類型的channel,如果對channel稍微有些了解的話,我們首先會想到,既然是返回一個channel,在調(diào)用Tick方法的過程中,必然創(chuàng)建了goroutine,該Goroutine負(fù)責(zé)發(fā)送數(shù)據(jù),喚醒被阻塞的定時任務(wù)。我在閱讀源碼之后,確實(shí)發(fā)現(xiàn)函數(shù)中g(shù)o出去了一個協(xié)程,處理定時任務(wù)。

按照當(dāng)前的理解,使用一個tick,需要go出去一個協(xié)程,效率和對內(nèi)存空間的占用肯定不能比sleep函數(shù)強(qiáng)。我們需要繼續(xù)閱讀源碼才拿獲取到真理。

簡單的調(diào)用過程我就不陳述了,我在這介紹一下核心結(jié)構(gòu)體和方法(刪除了部分判斷代碼,解釋我寫在表格中):

func (tb *timersBucket) addtimerLocked(t *timer) {
   t.i = len(tb.t)  //計算timersBucket中,當(dāng)前定時任務(wù)的長度
   tb.t = append(tb.t, t)// 將當(dāng)前定時任務(wù)加入timersBucket
   siftupTimer(tb.t, t.i)  //維護(hù)一個timer結(jié)構(gòu)體的最小堆(四叉樹),排序關(guān)鍵字為執(zhí)行時間,即該定時任務(wù)下一次執(zhí)行的時間
   if !tb.created {
      tb.created = true
      go timerproc(tb)// 如果還沒有創(chuàng)建過管理定時任務(wù)的協(xié)程,則創(chuàng)建一個,執(zhí)行通知管理timer的協(xié)程,最核心代碼
   }
}

timersBucket,顧名思義,時間任務(wù)桶,是外界不可見的全局變量。每當(dāng)有新的timer定時器任務(wù)時,會將timer加入到timersBucket中的timer切片。timerBucket結(jié)構(gòu)體如下:

type timersBucket struct {
   lock         mutex //添加新定時任務(wù)時需要加鎖(沖突點(diǎn)在于維護(hù)堆)
   t            []*timer //timer切片,構(gòu)造方式為四叉樹最小堆
}

func timerproc(tb *timersBucket) 詳細(xì)介紹

可以稱之為定時任務(wù)處理器,所有的定時任務(wù)都會加入timersBucket,然后在該函數(shù)中等待被處理。

等待被處理的timer,根據(jù)when字段(任務(wù)執(zhí)行的時間,int類型,納秒級別)構(gòu)成一個最小堆,每次處理完成堆頂?shù)哪硞€timer時,會給它的when字段加上定時任務(wù)循環(huán)間隔時間(即Tick(d Duration) 中的d參數(shù)),然后重新維護(hù)堆,保證when最小的timer在堆頂。當(dāng)堆中沒有可以處理的timer(有timer,但是還不到執(zhí)行時間),需要計算當(dāng)前時間和堆頂中timer的任務(wù)執(zhí)行時間差值delta,定時任務(wù)處理器沉睡delta段時間,等待被調(diào)度器喚醒。

核心代碼如下(注釋寫在每行代碼的后面,刪除一些判斷代碼以及不利于閱讀的非核心代碼):

func timerproc(tb *timersBucket) {
   for {
      lock(tb.lock) //加鎖
      now := nanotime()  //當(dāng)前時間的納秒值
      delta := int64(-1)  //最近要執(zhí)行的timer和當(dāng)前時間的差值
      for {
         if len(tb.t) == 0 {
            delta = -1
            break
         }//當(dāng)前無可執(zhí)行timer,直接跳出該循環(huán)
         t := tb.t[0]
         delta = t.when - now //取when組小的的timer,計算于當(dāng)前時間的差值
         if delta > 0 {
            break
         }// delta大于0,說明還未到發(fā)送channel時間,需要跳出循環(huán)去睡眠delta時間
         if t.period > 0 {
            // leave in heap but adjust next time to fire
            t.when += t.period * (1 + -delta/t.period)// 計算該timer下次執(zhí)行任務(wù)的時間
            siftdownTimer(tb.t, 0) //調(diào)整堆
         } else {
            // remove from heap,如果沒有設(shè)定下次執(zhí)行時間,則將該timer從堆中移除(time.after和time.sleep函數(shù)即是只執(zhí)行一次定時任務(wù))
            last := len(tb.t) - 1
            if last > 0 {
               tb.t[0] = tb.t[last]
               tb.t[0].i = 0
            }
            tb.t[last] = nil
            tb.t = tb.t[:last]
            if last > 0 {
               siftdownTimer(tb.t, 0)
            }
            t.i = -1 // mark as removed
         }
         f := t.f
         arg := t.arg
         seq := t.seq
         unlock(tb.lock)//解鎖
         f(arg, seq) //在channel中發(fā)送time結(jié)構(gòu)體,喚醒阻塞的協(xié)程
         lock(tb.lock)
      }
      if delta  0  {
         // No timers left - put goroutine to sleep.
         goparkunlock(tb.lock, "timer goroutine (idle)", traceEvGoBlock, 1)
         continue
      }// delta小于0說明當(dāng)前無定時任務(wù),直接進(jìn)行阻塞進(jìn)行睡眠
      tb.sleeping = true
      tb.sleepUntil = now + delta
      unlock(tb.lock)
      notetsleepg(tb.waitnote, delta)  //睡眠delta時間,喚醒之后就可以執(zhí)行在堆頂?shù)亩〞r任務(wù)了
   }
}

至此,time.Tick函數(shù)涉及到的主要功能就講解結(jié)束了,總結(jié)一下就是啟動定時任務(wù)時,會創(chuàng)建一個唯一協(xié)程,處理timer,所有的timer都在該協(xié)程中處理。

然后,我們再閱讀一下sleep的源碼實(shí)現(xiàn),核心源碼如下:

//go:linkname timeSleep time.Sleep
func timeSleep(ns int64) {
   *t = timer{} //創(chuàng)建一個定時任務(wù)
   t.when = nanotime() + ns //計算定時任務(wù)的執(zhí)行時間點(diǎn)
   t.f = goroutineReady //執(zhí)行方法
   tb.addtimerLocked(t)  //加入timer堆,并在timer定時任務(wù)執(zhí)行協(xié)程中等待被執(zhí)行
   goparkunlock(tb.lock, "sleep", traceEvGoSleep, 2) //睡眠,等待定時任務(wù)協(xié)程通知喚醒
}

讀了sleep的核心代碼之后,是不是突然發(fā)現(xiàn)和Tick函數(shù)的內(nèi)容很類似,都創(chuàng)建了timer,并加入了定時任務(wù)處理協(xié)程。神奇之處就在于,實(shí)際上這兩個函數(shù)產(chǎn)生的timer都放入了同一個timer堆,都在定時任務(wù)處理協(xié)程中等待被處理。

優(yōu)劣性對比,使用建議

現(xiàn)在我們知道了,Tick,Sleep,包括time.After函數(shù),都使用的timer結(jié)構(gòu)體,都會被放在同一個協(xié)程中統(tǒng)一處理,這樣看起來使用Tick,Sleep并沒有什么區(qū)別。

實(shí)際上是有區(qū)別的,Sleep是使用睡眠完成定時任務(wù),需要被調(diào)度喚醒。Tick函數(shù)是使用channel阻塞當(dāng)前協(xié)程,完成定時任務(wù)的執(zhí)行。當(dāng)前并不清楚golang 阻塞和睡眠對資源的消耗會有什么區(qū)別,這方面不能給出建議。

但是使用channel阻塞協(xié)程完成定時任務(wù)比較靈活,可以結(jié)合select設(shè)置超時時間以及默認(rèn)執(zhí)行方法,而且可以設(shè)置timer的主動關(guān)閉,以及不需要每次都生成一個timer(這方面節(jié)省系統(tǒng)內(nèi)存,垃圾收回也需要時間)。

所以,建議使用time.Tick完成定時任務(wù)。

補(bǔ)充:Golang 定時器timer和ticker

兩種類型的定時器:ticker和timer。兩者有什么區(qū)別呢?請看如下代碼:

ticker

package main
import (
        "fmt"
        "time"
)
func main() {
        d := time.Duration(time.Second*2)
        t := time.NewTicker(d)
        defer t.Stop()
        for {
                - t.C
                fmt.Println("timeout...")
        }
}

output:

timeout…

timeout…

timeout…

解析

ticker只要定義完成,從此刻開始計時,不需要任何其他的操作,每隔固定時間都會觸發(fā)。

timer

package main
import (
        "fmt"
        "time"
)
func main() {
        d := time.Duration(time.Second*2)
        t := time.NewTimer(d)
        defer t.Stop()
        for {
                - t.C
                fmt.Println("timeout...")
  // need reset
  t.Reset(time.Second*2)
        }
}

output:

timeout…

timeout…

timeout…

解析

使用timer定時器,超時后需要重置,才能繼續(xù)觸發(fā)。

ticker 例子展示

package main
import (
        "fmt"
        "time"
)
func main() {
        t := time.NewTicker(3*time.Second)
        defer t.Stop()
        fmt.Println(time.Now())
        time.Sleep(4*time.Second)
        for {
                select {
                case -t.C:
                        fmt.Println(time.Now())
                }
        }
}

output:

2018-04-02 19:08:22.2797 +0800 CST

2018-04-02 19:08:26.3087 +0800 CST

2018-04-02 19:08:28.2797 +0800 CST

2018-04-02 19:08:31.2797 +0800 CST

2018-04-02 19:08:34.2797 +0800 CST

以上為個人經(jīng)驗(yàn),希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。

您可能感興趣的文章:
  • 解決Golang time.Parse和time.Format的時區(qū)問題
  • 解決golang時間字符串轉(zhuǎn)time.Time的坑
  • golang的時區(qū)和神奇的time.Parse的使用方法
  • 對Golang中的runtime.Caller使用說明
  • Golang中的time.Duration類型用法說明
  • golang time包做時間轉(zhuǎn)換操作
  • golang xorm及time.Time自定義解決json日期格式的問題
  • golang time常用方法詳解

標(biāo)簽:汕頭 梅河口 銅川 雞西 吐魯番 蘭州 欽州 重慶

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《golang 定時任務(wù)方面time.Sleep和time.Tick的優(yōu)劣對比分析》,本文關(guān)鍵詞  golang,定時,任務(wù),方面,time.Sleep,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《golang 定時任務(wù)方面time.Sleep和time.Tick的優(yōu)劣對比分析》相關(guān)的同類信息!
  • 本頁收集關(guān)于golang 定時任務(wù)方面time.Sleep和time.Tick的優(yōu)劣對比分析的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    午夜视频在线观看一区二区三区| 91精品人妻一区二区三区果冻| 成人mm视频在线观看| av先锋影院| 久热精品视频在线免费观看| 亚洲柠檬福利资源导航| 三级小说欧洲区亚洲区| 欧美午夜精品久久久久久孕妇| 成人禁在线观看网站| 亚洲精品喷潮一区二区三区| 黄色一级视频免费观看| 亚洲黄色视屏| 婷婷色一区二区三区| 国产亚洲欧美激情| 欧美精品一区二区三区精品| 亚洲不卡在线视频| 视频福利一区| 欧美美女搞黄| 日本黄网站色大片免费观看| 少妇愉情理伦三级| 成人黄色av网址| 成人亚洲欧美日韩在线观看| 欧美日韩一区二区三区四区在线观看| 国产性xxxx18免费观看视频| 西野翔中文久久精品国产| www.com.cn成人| 日本欧美中文字幕| 7m第一福利500精品视频| 在线观看一二三区| 欧美最猛性xxxxx(亚洲精品)| 91精品国产高清一区二区三区蜜臀| 亚洲欧美偷拍另类| 日韩精品免费一区二区三区| 亚洲精品女人久久久| 亚洲成人av在线| 在线观看视频色潮| 性xxxxx| avtt天堂资源网站| 日韩成人在线电影网| 中文字幕在线1| 欧美6一10sex性hd| 国产日产欧美精品| 欧美日韩视频| 精品美女一区二区| 欧美一区国产在线| 亚洲欧美日韩一区二区三区四区| 国产精品综合av一区二区国产馆| 欧美韩国日本| 天堂av一区| 啊v在线视频| 欧美变态网站| 伊人中文字幕在线观看| 男人打飞机网站| 999热视频在线观看| 丰满人妻一区二区三区53号| 国产又粗又爽| 在线欧美成人| 中文字幕乱视频| 色哟哟在线观看| 欧美三日本三级少妇三2023| 开心激情综合| 亚洲国产精品久久| 国产成人av| 中文字幕欧美色图| 美腿丝袜在线亚洲一区| 最近最好的中文字幕2019免费| 欧美日韩在线播放| 福利一区和二区| 日韩欧美亚洲一区二区三区| 欧美黄网在线观看| 亚洲黄页网站| 少妇一级淫片免费放中国| 2018高清国产日本一道国产| 2021亚洲天堂| 亚洲成人黄色影院| 婷婷丁香久久| 中日韩男男gay无套| 色婷婷久久久亚洲一区二区三区| 三级在线观看视频| 在线国产三级| 亚洲精品在线视频观看| 欧美日韩天天操| 又黄又骚的视频| 中文字幕日韩一区| 性做久久久久久久免费看| 蜜桃传媒av| 亚洲天堂电影在线观看| 催眠调教后宫乱淫校园| 亚洲福利合集| 国产精品自拍一区| 亚洲欧洲一区二区三区久久| 国产波霸爆乳一区二区| 成人午夜又粗又硬又大| 99久久久久成人国产免费| 91亚洲精品久久久蜜桃网站| 国产精品日产欧美久久久久| 模特精品在线| 日本特黄特色aaa大片免费| 久久人人97超碰人人澡爱香蕉| 国产亚洲精品久久久久久牛牛| 亚洲精品乱码久久久久久蜜桃麻豆| 国产福利在线免费| 国产亚洲精品久久久久久青梅| 国产亚洲综合视频| 亚洲精品第五页| 欧美交a欧美精品喷水| 国产手机视频一区二区| 欧美成人免费全部| 午夜精品久久久久久久99热浪潮| 麻豆精品密在线观看| 亚洲国产日韩精品在线| 天天干夜夜干| 四虎精品在永久在线观看| 国产精品19乱码一区二区三区| 国产乱码精品1区2区3区| 久久精品在线视频| 国产精品亲子乱子伦xxxx裸| 国产激情91久久精品导航| 国产精品video| 天堂在线资源视频| 中文字幕欧美精品在线| 黄视频在线观看www| 日本一区二区三区播放| 青草视频在线观看视频| 久久国产三级精品| 在线综合亚洲欧美在线视频| 欧美另类高清videos的特点| 精品成人免费视频| 黄网站视频在线观看| 97人妻人人揉人人躁人人| 久久久久久欧美| 国产毛片毛片毛片毛片毛片毛片| 台湾佬中文娱乐久久久| 在线观看网站免费入口在线观看国内| 噜噜噜噜噜在线视频| 日韩欧美亚洲一区二区| 国内自拍欧美| 日本一二三不卡视频| 亚洲丁香久久久| 日韩欧美色电影| 亚洲天堂网一区| 免费拍拍拍网站| 国产91在线观看丝袜| 亚洲视频在线二区| 精品国产鲁一鲁****| 欧美精品一区二区三区在线四季| 性生活在线视频| 精品国产欧美日韩一区二区三区| 狠狠色狠狠色综合日日tαg| 一区二区乱子伦在线播放| 欧美午夜精品一区二区三区电影| 色视频精品视频在线观看| 国产片在线播放| 女教师淫辱の教室蜜臀av软件| av网在线观看| 制服.丝袜.亚洲.中文.综合懂色| 亚洲精品成人在线| 国内久久视频| 日韩精品一级中文字幕精品视频免费观看| 亚洲精品国产成人av在线| 3d动漫一区二区三区| 成人同人动漫免费观看| 欧美一a一片一级一片| 痴汉一区二区三区| 日本老妇乱子| 亚洲精品日产精品乱码不卡| 国产乱女淫av麻豆国产| av天堂一区二区三区| 色18美女社区| 国内精品美女在线观看| 亚洲成人网av| 91n在线视频| 深夜在线视频| 亚洲黄色a级片| 欧美日产国产成人免费图片| 亚洲日本精品视频| 污免费在线观看| 日本免费不卡| 麻豆免费网站| 日本成片免费高清| 亚洲午夜久久| 亚洲精品国产精品国自产| 一区三区自拍| 亚洲尤物影院| 久久偷窥视频| 日本大片在线观看| 福利成人在线观看| 在线播放精品视频| 久久综合色播五月| 亚洲永久免费| 精品国产一区久久| 国产性70yerg老太| 91精品在线观看国产| 福利一区和二区| www插插插无码免费视频网站| 成人盗摄视频| 欧美在线看片a免费观看| 欧美全黄视频| 呻吟揉丰满对白91乃国产区| 亚洲色图美腿丝袜| 欧美视频免费在线观看| av在线免费播放网址| 久久久久久久久综合影视网| 亚洲国产精品999| 精品少妇一区二区三区免费观看| 成人动漫免费在线观看| 欧美一级特黄aaaaaa大片在线观看| 久久久久久毛片免费看| 日本视频网址| 久久久精品99| 亚洲人成在线播放| 亚洲一区二区三区免费在线观看| 色国产精品一区在线观看| 欧美一区二区私人影院日本| 最新高清无码专区| 4438x成人网最大色成网站| 亚洲第一精品在线观看| 欧美一区亚洲一区| 成人在线观看网址| 国产精品第1页| 国产精品欧美久久久久无广告| 污视频网站在线免费| 欧美日韩一级二级三级| 成人激情小说乱人伦| 欧美精品中文字幕亚洲专区| 日韩精品一区国产| 国产亚洲一区精品| 中文字幕日韩电影| 久草国产精品视频| 国产性生交xxxxx免费| 国产日韩欧美在线观看| 青青视频在线播放| 亚洲欧美久久婷婷爱综合一区天堂| 亚洲国产福利视频| www免费在线观看| 精品在线观看一区| 国产一级一片免费播放| 欧美亚洲精品在线观看| 国产专区在线| 国产69精品久久久久孕妇| 欧美1区二区| 中文字幕欧美精品日韩中文字幕| 欧美国产一区在线| 黄色电影免费在线观看| 黄色三及免费看| av一区二区三区在线| 极品束缚调教一区二区网站| 乱子伦在线视频| 久久99国产精品久久久久久久久| 亚洲欧洲美洲国产香蕉| 精品国产一区二区三区忘忧草| 任你弄精品视频免费观看| 国产一区欧美一区| 国产黄视频在线观看| 日韩av手机在线播放| 精品久久久三级丝袜| а天堂8中文最新版在线官网| 少妇欧美激情一区二区三区| 在线观看中文字幕| 免费看日产一区二区三区| 成人免费观看49www在线观看| 中文字幕久热精品在线视频| 又黄又免费的网站| 高清成人在线| 久久噜噜噜精品国产亚洲综合| 中国人体摄影一区二区三区| 国产精品一区二区免费看| 成人免费a视频| 精品久久免费视频| 国产真实乱偷精品视频免| 鲁大师私人影院在线观看| 午夜午夜精品一区二区三区文| 99re在线视频播放| 国产日韩亚洲欧美精品| 亚洲国产成人久久| 麻豆91蜜桃| 黄色精品视频| 成人a区在线观看| 日韩一区自拍| 91玉足脚交嫩脚丫在线播放| 色综合色综合色综合| 国产欧美一区二区精品性色超碰| 久久亚洲精品毛片| 国产精品三级久久久久三级| 黄色污网站在线观看| 日韩美女在线视频| 国产成人综合一区二区三区| 日本成人片在线| 男人晚上看的视频| 青草在线视频| 黄大色黄女片18免费| 一区二区日本视频| 狠狠久久亚洲欧美专区| 蜜桃tv一区二区三区| 日韩电影免费网站| 精品久久无码中文字幕| 91一区二区三区在线播放| 青青草99啪国产免费| 国产视频网站一区二区三区| caoporn视频在线| 国产在线国偷精品免费看| 国产3级在线观看| 制服丝袜中文字幕亚洲| 欧美专区第一页| 精品综合久久88少妇激情| 一卡二卡三卡四卡| 99re热视频| 蜜月aⅴ免费一区二区三区| 色视频成人在线观看免| 亚洲va在线va天堂成人| 中文字幕在线精品| 大地资源二中文在线影视观看| 麻豆精品视频在线观看| 久久久久久亚洲| 成人中文字幕在线播放| 亚洲午夜一区二区三区| 黄色av网站在线观看| 亚洲黄色片免费| 成人444kkkk在线观看| 首页欧美精品中文字幕| 亚洲精品永久免费视频| 国产综合欧美| 日韩不卡视频在线观看| av网站在线免费看推荐| 一区不卡在线观看| 国产中文字幕在线播放| 亚洲国产一区二区三区在线| 又大又长粗又爽又黄少妇视频|