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

Golang控制協(xié)程執(zhí)行順序方法詳解

 更新時(shí)間:2022年11月18日 15:06:29   作者:Mingvvv  
這篇文章主要介紹了Golang控制協(xié)程執(zhí)行順序的方法,Golang的語(yǔ)法和運(yùn)行時(shí)直接內(nèi)置了對(duì)并發(fā)的支持。Golang里的并發(fā)指的是能讓某個(gè)函數(shù)獨(dú)立于其他函數(shù)運(yùn)行的能力

在 Go 里面的協(xié)程執(zhí)行實(shí)際上默認(rèn)是沒(méi)有嚴(yán)格的先后順序的。由于 Go 語(yǔ)言 GPM 模型的設(shè)計(jì)理念,真正執(zhí)行實(shí)際工作的實(shí)際上是 GPM 中的 M(machine) 執(zhí)行器,而我們的協(xié)程任務(wù) G(goroutine) 協(xié)程需要被 P(produce) 關(guān)聯(lián)到某個(gè) M 上才能被執(zhí)行。而每一個(gè) P 都有一個(gè)私有隊(duì)列,除此之外所有的 P 還共用一個(gè)公共隊(duì)列。因此當(dāng)我們創(chuàng)建了一個(gè)協(xié)程之后,并不是立即執(zhí)行,而是進(jìn)入隊(duì)列等待被分配,且不同隊(duì)列之間沒(méi)有順序關(guān)系可言。

但是在有些時(shí)候,我們并不是希望所有的協(xié)程都隨機(jī)執(zhí)行,所以我們需要想辦法控制協(xié)程的執(zhí)行順序,這里整理了幾種控制協(xié)程執(zhí)行順序的方法。

循環(huán)控制

思路就是我們要給每一個(gè)子協(xié)程設(shè)置一個(gè)序號(hào),當(dāng)前一個(gè)序號(hào)的協(xié)程執(zhí)行完之后,才能執(zhí)行下一個(gè)。

所以我們需要一個(gè)公共變量去記錄當(dāng)前可以執(zhí)行的協(xié)程的序號(hào),同時(shí)這個(gè)變量必須是線程安全的,以確保對(duì)于每個(gè)協(xié)程的每一次讀寫操作都是正確的。

首先循環(huán)等待合適的時(shí)機(jī):

這個(gè)函數(shù)會(huì)不斷循環(huán)獲取一個(gè) count 值,當(dāng) count 的值和參中的 i 相同時(shí),他就會(huì)進(jìn)入執(zhí)行參數(shù) fn 代表的函數(shù),并且將 count 的值 +1 。

否則它將等待一納秒然后重復(fù)以上步驟。

var count uint32
func sequence(i uint32, fn func()) {
	for {
		//使用原子操作
		if n := atomic.LoadUint32(&count); n == i {
			fn()
			atomic.AddUint32(&count, 1)
			break
		}
		time.Sleep(time.Nanosecond)
	}
}

然后用 sequence 來(lái)控制協(xié)程順序:

我們將要執(zhí)行的邏輯放在函數(shù) fn 中,并放在 sequence 函數(shù)中執(zhí)行,由函數(shù) sequence 去確保寫成的執(zhí)行順序。

最后 sequence(times, func() {}) 是為了讓主協(xié)程最后退出,當(dāng)然我們可一個(gè)使用通道 chan 去實(shí)現(xiàn)(可以參考上一篇)。

func main() {
	var times uint32 = 5
	for i := uint32(0); i < times; i++ {
		go func(i uint32) {
			fn := func() {
				fmt.Printf("this i is %v\n", i)
			}
			sequence(i, fn)
		}(i)
	}
	//讓主協(xié)程等待最后執(zhí)行
	sequence(times, func() {})
}

執(zhí)行結(jié)果:

this i is 0
this i is 1
this i is 2
this i is 3
this i is 4

通道控制

原理就是,前后協(xié)程之間通過(guò)通道去相互限制,后一個(gè)協(xié)程嘗試去獲取一個(gè)通道里面的值,當(dāng)通道中沒(méi)有值時(shí),就會(huì)一直阻塞。

而前一個(gè)協(xié)程則負(fù)責(zé)關(guān)閉通道,或向通道中發(fā)送值,當(dāng)前一個(gè)協(xié)程完成了這個(gè)操作,后一個(gè)協(xié)程才可以結(jié)束阻塞,繼續(xù)執(zhí)行。

func main() {
	c1 := make(chan struct{})
	c2 := make(chan struct{})
	c3 := make(chan struct{})
	go func() {
		//協(xié)程一 不受限制 直接執(zhí)行 執(zhí)行結(jié)束后關(guān)閉通道一
		fmt.Println("this value is 0")
		close(c1)
	}()
	go func() {
		//協(xié)程二 需要從通道一中接收值 ,或者通道關(guān)閉時(shí),獲取到接收失敗的結(jié)果,否則一直阻塞
		//執(zhí)行結(jié)束后關(guān)閉通道二
		<-c1
		fmt.Println("this value is 1")
		close(c2)
	}()
	go func() {
		//協(xié)程三 需要從通道二中接收值 ,或者通道關(guān)閉時(shí),獲取到接收失敗的結(jié)果,否則一直阻塞
		//執(zhí)行結(jié)束后關(guān)閉通道三
		<-c2
		fmt.Println("this value is 2")
		close(c3)
	}()
	//主協(xié)程 需要從通道三中接收值 ,或者通道關(guān)閉時(shí),獲取到接收失敗的結(jié)果,否則一直阻塞
	<-c3
}

執(zhí)行結(jié)果

this value is 0
this value is 1
this value is 2

互斥鎖 async.Mutex

直接上代碼

func main() {
	times := 5
	//創(chuàng)建一個(gè)互斥鎖數(shù)組 多一個(gè)給主協(xié)程用
	var cc = make([]*sync.Mutex, times+1)
	//往數(shù)組中塞入互斥鎖,默認(rèn)直接加鎖
	for i := 0; i < len(cc); i++ {
		m := &sync.Mutex{}
		m.Lock()
		cc[i] = m
	}
	for i := 0; i < times; i++ {
		//創(chuàng)建子協(xié)程
		go func(index int) {
			//子協(xié)程嘗試為數(shù)組中對(duì)應(yīng) index 位置的鎖加鎖,獲取不到鎖就等待
			//因?yàn)槌跏蓟倪@些互斥鎖默認(rèn)就已經(jīng)被鎖住了,所以這里創(chuàng)建的子協(xié)程都會(huì)被阻塞
			//一旦獲取到鎖,就執(zhí)行邏輯,最后將當(dāng)前index的鎖和index+1的鎖釋放,這樣正在等待 index +1 位置的鎖的子協(xié)程就可以繼續(xù)執(zhí)行了
			cc[index].Lock()
			fmt.Printf("this value is %d \n", index)
			cc[index].Unlock()
			cc[index+1].Unlock()
		}(i)
	}
	//將index 為 0 位置的鎖解鎖,讓第一個(gè)子協(xié)程可以繼續(xù)執(zhí)行
	cc[0].Unlock()
	//為 index 為 times 的鎖加鎖,只有當(dāng)最后一個(gè)子協(xié)程執(zhí)行完畢后,這個(gè)鎖才會(huì)解鎖,主協(xié)程才能繼續(xù)向下走
	cc[times].Lock()
	cc[times].Unlock()
}

到此這篇關(guān)于Golang控制協(xié)程執(zhí)行順序方法詳解的文章就介紹到這了,更多相關(guān)Go協(xié)程執(zhí)行順序內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • go語(yǔ)言 xorm框架 postgresql 的用法及詳細(xì)注解

    go語(yǔ)言 xorm框架 postgresql 的用法及詳細(xì)注解

    這篇文章主要介紹了go語(yǔ)言 xorm框架 postgresql 的用法及詳細(xì)注解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-12-12
  • Go?Excelize?API源碼閱讀SetSheetViewOptions示例解析

    Go?Excelize?API源碼閱讀SetSheetViewOptions示例解析

    這篇文章主要為大家介紹了Go-Excelize?API源碼閱讀SetSheetViewOptions示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-08-08
  • Golang中map數(shù)據(jù)類型的使用方法

    Golang中map數(shù)據(jù)類型的使用方法

    這篇文章主要介紹了Golang中map數(shù)據(jù)類型的使用方法,文章圍繞主題展開(kāi)詳細(xì)的內(nèi)容戒殺,具有一定的參考價(jià)值,需要的朋友可以參考一下
    2022-09-09
  • Go語(yǔ)言中的并發(fā)模式你了解了嗎

    Go語(yǔ)言中的并發(fā)模式你了解了嗎

    工作中查看項(xiàng)目代碼,發(fā)現(xiàn)會(huì)存在使用?GO?語(yǔ)言做并發(fā)的時(shí)候出現(xiàn)各種各樣的異常情況,實(shí)際上,出現(xiàn)上述的情況,還是因?yàn)槲覀儗?duì)于?GO?語(yǔ)言的并發(fā)模型和涉及的?GO?語(yǔ)言基礎(chǔ)不夠扎實(shí),所以本文小編就來(lái)帶大家深入了解下Go語(yǔ)言中的并發(fā)模式吧
    2023-08-08
  • Go基于struct?tag實(shí)現(xiàn)結(jié)構(gòu)體字段級(jí)別的訪問(wèn)控制

    Go基于struct?tag實(shí)現(xiàn)結(jié)構(gòu)體字段級(jí)別的訪問(wèn)控制

    本文將會(huì)基于這個(gè)主題展開(kāi),討論Go中的結(jié)構(gòu)體tag究竟是什么,我們?cè)撊绾卫盟?另外,文末還提供了一個(gè)實(shí)際案例,實(shí)現(xiàn)結(jié)構(gòu)體字段級(jí)別的訪問(wèn),幫助我們進(jìn)一步提升對(duì)struct tag的理解
    2024-02-02
  • Golang接口使用教程詳解

    Golang接口使用教程詳解

    在?Go?語(yǔ)言中接口包含兩種含義:它既是方法的集合,?同時(shí)還是一種類型并且在Go?語(yǔ)言中是隱式實(shí)現(xiàn)的。本文通過(guò)示例詳細(xì)介紹了Golang接口的使用,需要的可以參考一下
    2022-09-09
  • Golang Web 框架Iris安裝部署

    Golang Web 框架Iris安裝部署

    這篇文章主要為大家介紹了Golang Web 框架Iris安裝部署,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-08-08
  • Go語(yǔ)言使用Gob傳輸數(shù)據(jù)

    Go語(yǔ)言使用Gob傳輸數(shù)據(jù)

    本文主要介紹了Go語(yǔ)言使用Gob傳輸數(shù)據(jù),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-04-04
  • Go語(yǔ)言的匿名字段實(shí)現(xiàn)組合復(fù)用實(shí)例探究

    Go語(yǔ)言的匿名字段實(shí)現(xiàn)組合復(fù)用實(shí)例探究

    這篇文章主要為大家介紹了Go語(yǔ)言的匿名字段實(shí)現(xiàn)組合復(fù)用實(shí)例探究,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2024-01-01
  • 重學(xué)Go語(yǔ)言之如何使用Modules

    重學(xué)Go語(yǔ)言之如何使用Modules

    Go語(yǔ)言在Go.1.11版本發(fā)布了Go?Modules,這是一種新的Go項(xiàng)目依賴管理解決方案,可以讓Go項(xiàng)目的依賴包關(guān)系更加清晰,也更容易管理,下面就來(lái)看看Modules是如何使用的吧
    2023-07-07

最新評(píng)論