欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺析go語言如何實(shí)現(xiàn)協(xié)程的搶占式調(diào)度的

 更新時(shí)間:2024年04月23日 08:58:37   作者:shark_chili  
go語言通過GMP模型實(shí)現(xiàn)協(xié)程并發(fā),為了避免單協(xié)程持續(xù)持有線程導(dǎo)致線程隊(duì)列中的其他協(xié)程饑餓問題,設(shè)計(jì)者提出了一個(gè)搶占式調(diào)度機(jī)制,本文會(huì)基于一個(gè)簡(jiǎn)單的代碼示例對(duì)搶占式調(diào)度過程進(jìn)行深入講解剖析

詳解協(xié)程搶占式調(diào)度

函數(shù)調(diào)用間進(jìn)行搶占式調(diào)度

假設(shè)我們現(xiàn)在有這樣一個(gè)協(xié)程,它會(huì)進(jìn)行函數(shù)嵌套調(diào)用,代碼如下所示:

func foo1() {
 fmt.Println("foo1調(diào)用foo2")
 foo2()
}

func foo2() {
 fmt.Println("foo2調(diào)用foo3")
 foo3()
}

func foo3() {
 fmt.Println("foo3")
}

func main() {
 //設(shè)置WaitGroup等待協(xié)程運(yùn)行結(jié)束
 var wg sync.WaitGroup
 wg.Add(1)
 //通過協(xié)程調(diào)用foo1
 go func() {
  defer wg.Done()
  foo1()
 }()
 //等待協(xié)程運(yùn)行結(jié)束
 wg.Wait()
}

我們給出運(yùn)行結(jié)果:

foo1調(diào)用foo2
foo2調(diào)用foo3
foo3

基于這段代碼示例,我們通過這段指令獲取plan9匯編碼:

go build -gcflags -S main.go

可以看到在foo1插入runtime.morestack_noctxt方法,該方法是用于檢查當(dāng)前協(xié)程是否有足夠的堆棧空間以保證函數(shù)的正常調(diào)用,基于這一點(diǎn),go就會(huì)在進(jìn)行這部檢查時(shí)順帶檢查協(xié)程的執(zhí)行時(shí)長(zhǎng),一旦超過10ms該方法就會(huì)將協(xié)程設(shè)置為標(biāo)記可被搶占:

 0x0061 00097 (F:\github\test\main.go:8) CALL    runtime.morestack_noctxt(SB)

如下圖,我們的調(diào)用的函數(shù)都會(huì)被插入一個(gè)morestack通過這個(gè)標(biāo)記判斷當(dāng)前協(xié)程執(zhí)行耗時(shí),一旦發(fā)現(xiàn)超過10ms則會(huì)直接通過搶占式調(diào)度的方法g0協(xié)程直接調(diào)用schedule方法獲取另外的協(xié)程進(jìn)行調(diào)用:

這一點(diǎn)我們可以在asm_amd64.s看到morestacknewstack的代碼,而newstack就是實(shí)現(xiàn)搶占式調(diào)度的核心:

TEXT runtime·morestack(SB),NOSPLIT,$0-0
 // Cannot grow scheduler stack (m->g0).
 get_tls(CX)
 MOVQ g(CX), BX
 MOVQ g_m(BX), BX
 MOVQ m_g0(BX), SI
 CMPQ g(CX), SI
 JNE 3(PC)
 CALL runtime·badmorestackg0(SB)
 CALL runtime·abort(SB)

    //......
    //函數(shù)調(diào)用前會(huì)調(diào)用newstack進(jìn)行搶占式的檢查
 CALL runtime·newstack(SB)
 CALL runtime·abort(SB) // crash if newstack returns
 RET

上述的newstack方法在stack.go中,如果當(dāng)前協(xié)程可被搶占則會(huì)調(diào)用gopreempt_m回到g0調(diào)用schedule方法從協(xié)程隊(duì)列中拿到新的協(xié)程執(zhí)行任務(wù):

func newstack() {
 preempt := stackguard0 == stackPreempt


 //如果preempt 為true,則直接當(dāng)前協(xié)程被標(biāo)記為搶占直接調(diào)用gopreempt_m讓出線程執(zhí)行權(quán)
 if preempt {
  if gp == thisg.m.g0 {
   throw("runtime: preempt g0")
  }
  //......

  // Act like goroutine called runtime.Gosched.
  gopreempt_m(gp) // never return
 }
}

基于系統(tǒng)調(diào)用發(fā)起信號(hào)的搶占式調(diào)度

假設(shè)我們的協(xié)程沒有進(jìn)行額外的函數(shù)調(diào)用,是否就意味著當(dāng)前協(xié)程的線程不能被搶占呢?很明顯不是這樣:

網(wǎng)絡(luò)傳輸過程中需要發(fā)送某些緊急消息希望通過已有連接迅速將消息通知給對(duì)端時(shí),就會(huì)產(chǎn)生SIGURG信號(hào),go語言就會(huì)在收到此信號(hào)時(shí)觸發(fā)搶占式調(diào)度。

進(jìn)行GC工作時(shí)像目標(biāo)線程發(fā)送信號(hào)由此實(shí)現(xiàn)搶占式調(diào)度。

對(duì)于第一點(diǎn)我們可以在signal_unix.gosighandler方法得以印證,可以看到它會(huì)判斷sig 是否為_SIGURG若是則調(diào)用doSigPreempt進(jìn)行搶占式調(diào)度

func sighandler(sig uint32, info *siginfo, ctxt unsafe.Pointer, gp *g) {
 
 //如果傳入的信號(hào)為_SIGURG則調(diào)用doSigPreempt回到schedule實(shí)現(xiàn)搶占式調(diào)度
 if sig == sigPreempt && debug.asyncpreemptoff == 0 && !delayedSignal {
  // Might be a preemption signal.
  doSigPreempt(gp, c)
  
 }
 //......
}

doSigPreempt會(huì)通過調(diào)用asyncPreempt最終執(zhí)行到preempt.goasyncPreempt2調(diào)用到和上文函數(shù)調(diào)用搶占式調(diào)度方法gopreempt_m回到schedule方法從而完成搶占式調(diào)度:

func doSigPreempt(gp *g, ctxt *sigctxt) {
 //......
 if wantAsyncPreempt(gp) {
  if ok, newpc := isAsyncSafePoint(gp, ctxt.sigpc(), ctxt.sigsp(), ctxt.siglr()); ok {
   // 調(diào)用asyncPreempt內(nèi)部會(huì)得到一個(gè)和上文函數(shù)調(diào)用時(shí)搶占式調(diào)度的方法gopreempt_m的調(diào)用從而回到schedule方法
   ctxt.pushCall(abi.FuncPCABI0(asyncPreempt), newpc)
  }
 }

 //......
}

到此這篇關(guān)于淺析go語言如何實(shí)現(xiàn)協(xié)程的搶占式調(diào)度的的文章就介紹到這了,更多相關(guān)go協(xié)程搶占式調(diào)度內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • golang判斷key是否在map中的代碼

    golang判斷key是否在map中的代碼

    這篇文章主要介紹了golang判斷key是否在map中的代碼,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2021-04-04
  • Go語言排序算法之插入排序與生成隨機(jī)數(shù)詳解

    Go語言排序算法之插入排序與生成隨機(jī)數(shù)詳解

    從這篇文章開始將帶領(lǐng)大家學(xué)習(xí)Go語言的經(jīng)典排序算法,比如插入排序、選擇排序、冒泡排序、希爾排序、歸并排序、堆排序和快排,二分搜索,外部排序和MapReduce等,本文將先詳細(xì)介紹插入排序,并給大家分享了go語言生成隨機(jī)數(shù)的方法,下面來一起看看吧。
    2017-11-11
  • go語言中Timer和Ticker兩種計(jì)時(shí)器的使用

    go語言中Timer和Ticker兩種計(jì)時(shí)器的使用

    go語言中有Timer和Ticker這樣的兩種計(jì)時(shí)器,兩種計(jì)時(shí)器分別實(shí)現(xiàn)了不同的計(jì)時(shí)功能,本文主要介紹了go語言中Timer和Ticker兩種計(jì)時(shí)器的使用,感興趣的可以了解一下
    2024-08-08
  • Go語言配置解析庫(kù)viper的使用指南

    Go語言配置解析庫(kù)viper的使用指南

    viper?配置管理解析庫(kù),是由大神?Steve?Francia?開發(fā),本文就來和大家詳細(xì)講講它的具體使用,文中的示例代碼講解詳細(xì),需要的可以收藏一下
    2023-06-06
  • Go中runtime.Caller的使用

    Go中runtime.Caller的使用

    這篇文章主要介紹了Go中runtime.Caller的使用,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧
    2024-03-03
  • Golang驗(yàn)證器之validator是使用詳解

    Golang驗(yàn)證器之validator是使用詳解

    Validator是一個(gè)?Golang?的第三方庫(kù),用于對(duì)數(shù)據(jù)進(jìn)行校驗(yàn),常用于?API?的開發(fā)中,對(duì)客戶端發(fā)出的請(qǐng)求數(shù)據(jù)進(jìn)行嚴(yán)格校驗(yàn),防止惡意請(qǐng)求。本文通過示例詳細(xì)講解了Validator的使用,需要的可以參考一下
    2022-08-08
  • Golang TCP網(wǎng)絡(luò)編程的具體實(shí)現(xiàn)

    Golang TCP網(wǎng)絡(luò)編程的具體實(shí)現(xiàn)

    go語言是一門功能強(qiáng)大的編程語言,它提供了眾多的網(wǎng)絡(luò)編程庫(kù),其中包括tcp/ip,本文主要介紹了Golang TCP網(wǎng)絡(luò)編程的具體實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以來了解一下
    2024-06-06
  • Golang Gob編碼(gob包的使用詳解)

    Golang Gob編碼(gob包的使用詳解)

    這篇文章主要介紹了Golang Gob編碼(gob包的使用詳解),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2021-05-05
  • Go語言學(xué)習(xí)之結(jié)構(gòu)體和方法使用詳解

    Go語言學(xué)習(xí)之結(jié)構(gòu)體和方法使用詳解

    這篇文章主要為大家詳細(xì)介紹了Go語言中結(jié)構(gòu)體和方法的使用,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Go語言有一定的幫助,需要的可以參考一下
    2022-04-04
  • Golang實(shí)現(xiàn)EasyCache緩存庫(kù)實(shí)例探究

    Golang實(shí)現(xiàn)EasyCache緩存庫(kù)實(shí)例探究

    這篇文章主要為大家介紹了Golang實(shí)現(xiàn)EasyCache緩存庫(kù)實(shí)例探究,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2024-01-01

最新評(píng)論