亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

4個場景教會你Go中Goroutine和通道是怎么用的

 更新時間:2023年05月26日 10:28:19   作者:不背鍋運維  
本篇給出了4個在運維開發(fā)工作中較為常見的且也是比較典型的場景,通過這些場景來帶大家掌握Go中Goroutine和通道是怎么使用的,需要的可以學(xué)習(xí)一下

開篇

這段時間把主要精力都放在了K8S上,差點把Golang給忘了。那本篇就分享一下并發(fā)相關(guān)的內(nèi)容(Goroutine和通道)。 本篇給出4個場景,這4個場景是在運維開發(fā)工作中較為常見的且也是比較典型的場景。通過這些代碼示例,讓你知道Goroutine和通道在運維開發(fā)中是怎么應(yīng)用的??偠灾?,言而總之,當(dāng)涉及到處理并發(fā)和并行任務(wù)時,Goroutine和通道是非常強悍的,可以讓我們開發(fā)出高效的、牛逼的并發(fā)程序。

實戰(zhàn)場景

1.并發(fā)執(zhí)行任務(wù)的場景

場景:假設(shè)需要編寫一個程序,用于批量執(zhí)行某個操作(例如部署應(yīng)用程序、更新配置等)到多臺服務(wù)器上。

供參考的代碼示例:

package?main

import?(
?"fmt"
?"sync"
)

//?服務(wù)器結(jié)構(gòu)體
type?Server?struct?{
?Name?string
?//?其他服務(wù)器相關(guān)的字段
}

//?執(zhí)行任務(wù)的函數(shù)
func?executeTask(server?Server,?task?string)?{
?//?連接服務(wù)器并執(zhí)行任務(wù)的邏輯
?fmt.Printf("執(zhí)行任務(wù)?[%s]?到服務(wù)器?[%s]\n",?task,?server.Name)
?//?執(zhí)行操作的代碼
}

func?main()?{
?//?服務(wù)器列表
?servers?:=?[]Server{
??{Name:?"Server1"},
??{Name:?"Server2"},
??{Name:?"Server3"},
??//?添加更多的服務(wù)器
?}

?//?任務(wù)列表
?tasks?:=?[]string{"部署應(yīng)用程序",?"更新配置",?"執(zhí)行命令",?"其他任務(wù)"}

?//?創(chuàng)建一個任務(wù)通道,用于發(fā)送任務(wù)到Goroutine池
?taskChannel?:=?make(chan?string)

?//?創(chuàng)建一個等待組,用于等待所有Goroutine執(zhí)行完畢
?var?wg?sync.WaitGroup
?wg.Add(len(servers))

?//?啟動Goroutine池
?for?_,?server?:=?range?servers?{
??go?func(server?Server)?{
???//?標(biāo)記任務(wù)完成時,通知等待組減少一個計數(shù)
???defer?wg.Done()

???//?從任務(wù)通道中接收任務(wù),并執(zhí)行
???for?task?:=?range?taskChannel?{
????executeTask(server,?task)
???}
??}(server)
?}

?//?將任務(wù)發(fā)送到任務(wù)通道
?for?_,?task?:=?range?tasks?{
??taskChannel?<-?task
?}

?//?關(guān)閉任務(wù)通道,表示所有任務(wù)都已發(fā)送
?close(taskChannel)

?//?等待所有Goroutine執(zhí)行完畢
?wg.Wait()
}

上面的代碼,創(chuàng)建了一個Goroutine池,每個Goroutine代表一臺服務(wù)器,通過通道將任務(wù)分發(fā)給Goroutine進行并發(fā)執(zhí)行。每個Goroutine負(fù)責(zé)連接到服務(wù)器,并執(zhí)行相應(yīng)的操作。這樣可以加速任務(wù)的執(zhí)行,同時提高資源利用率。

2.并發(fā)日志處理的場景

場景:假設(shè)需要將大量的日志數(shù)據(jù)并發(fā)地寫入到不同的目標(biāo)中(例如文件、數(shù)據(jù)庫、消息隊列等)。

供參考的代碼示例:

package?main

import?(
?"fmt"
?"log"
?"os"
?"sync"
)

//?日志結(jié)構(gòu)體
type?Log?struct?{
?Message?string
?//?其他日志相關(guān)的字段
}

func?main()?{
?//?創(chuàng)建一個日志通道,用于發(fā)送日志數(shù)據(jù)
?logChannel?:=?make(chan?Log)

?//?創(chuàng)建一個等待組,用于等待所有Goroutine執(zhí)行完畢
?var?wg?sync.WaitGroup

?//?啟動一個Goroutine處理日志寫入操作
?wg.Add(1)
?go?func()?{
??//?標(biāo)記日志寫入完畢時,通知等待組減少一個計數(shù)
??defer?wg.Done()

??//?打開文件進行日志寫入
??file,?err?:=?os.OpenFile("log.txt",?os.O_APPEND|os.O_CREATE|os.O_WRONLY,?0644)
??if?err?!=?nil?{
???log.Fatal(err)
??}
??defer?file.Close()

??//?創(chuàng)建一個日志寫入器
??logger?:=?log.New(file,?"",?log.LstdFlags)

??//?從日志通道中接收日志數(shù)據(jù),并寫入到目標(biāo)中
??for?log?:=?range?logChannel?{
???logger.Println(log.Message)
??}
?}()

?//?并發(fā)地向日志通道發(fā)送日志數(shù)據(jù)
?for?i?:=?0;?i?<?10;?i++?{
??wg.Add(1)
??go?func(index?int)?{
???//?標(biāo)記發(fā)送完日志數(shù)據(jù)時,通知等待組減少一個計數(shù)
???defer?wg.Done()

???//?構(gòu)造日志數(shù)據(jù)
???log?:=?Log{
????Message:?fmt.Sprintf("日志消息?%d",?index),
????//?設(shè)置其他日志字段
???}

???//?發(fā)送日志數(shù)據(jù)到日志通道
???logChannel?<-?log
??}(i)
?}

?//?關(guān)閉日志通道,表示所有日志數(shù)據(jù)都已發(fā)送
?close(logChannel)

?//?等待日志寫入操作的Goroutine執(zhí)行完畢
?wg.Wait()
}

在上面的代碼中,使用了一個專門的Goroutine來處理日志寫入操作,該Goroutine從一個日志通道中讀取日志數(shù)據(jù),并將其寫入到目標(biāo)中。其他的Goroutine可以并發(fā)地向該通道發(fā)送日志數(shù)據(jù),而不會因為寫入操作而阻塞。

3.異步任務(wù)調(diào)度的場景

在實際的運維工作中,有種場景是需要按照一定的調(diào)度策略異步執(zhí)行一些任務(wù)。比如這樣的場景:定期備份數(shù)據(jù)庫、定時清理無效數(shù)據(jù)等。

供參考的代碼示例:

package?main

import?(
?"fmt"
?"time"
)

//?執(zhí)行任務(wù)的函數(shù)
func?executeTask(task?string)?{
?//?執(zhí)行任務(wù)的邏輯
?fmt.Println("執(zhí)行任務(wù):",?task)
?//?具體的任務(wù)操作代碼
}

func?main()?{
?//?創(chuàng)建一個定時器,每隔一段時間觸發(fā)一次
?timer?:=?time.NewTimer(5?*?time.Second)

?//?啟動一個Goroutine等待定時器的觸發(fā)事件
?go?func()?{
??//?等待定時器的觸發(fā)事件
??<-timer.C
??//?定時器觸發(fā)后執(zhí)行任務(wù)
??executeTask("定時任務(wù)1")
??//?重新設(shè)置定時器,以實現(xiàn)循環(huán)調(diào)度
??timer.Reset(10?*?time.Second)
?}()

?//?主線程繼續(xù)執(zhí)行其他任務(wù)
?//?...

?//?阻塞主線程,保持程序運行
?select?{}
}

4.并發(fā)任務(wù)協(xié)作的場景

在某些情況下,你可能需要多個Goroutine之間進行協(xié)作和同步。例如,一個Goroutine負(fù)責(zé)生產(chǎn)任務(wù),而另一個Goroutine負(fù)責(zé)消費任務(wù)并進行處理。你可以使用通道來實現(xiàn)生產(chǎn)者-消費者模型。生產(chǎn)者將任務(wù)發(fā)送到通道中,而消費者從通道中接收任務(wù)并進行處理。通過這種方式,可以實現(xiàn)任務(wù)的有效分配和協(xié)同處理。

package?main

import?(
?"fmt"
?"time"
)

//?任務(wù)結(jié)構(gòu)體
type?Task?struct?{
?ID???int
?Data?string
?//?其他任務(wù)相關(guān)的字段
}

//?生產(chǎn)者,負(fù)責(zé)生產(chǎn)任務(wù)并發(fā)送到通道
func?producer(ch?chan<-?Task)?{
?for?i?:=?1;?i?<=?10;?i++?{
??task?:=?Task{
???ID:???i,
???Data:?fmt.Sprintf("任務(wù)?%d",?i),
???//?設(shè)置其他任務(wù)字段
??}
??ch?<-?task
??fmt.Println("生產(chǎn)任務(wù):",?task.Data)
??time.Sleep(500?*?time.Millisecond)?//?模擬生產(chǎn)任務(wù)的耗時
?}
?close(ch)?//?關(guān)閉通道,表示所有任務(wù)都已生產(chǎn)完畢
}

//?消費者,負(fù)責(zé)從通道接收任務(wù)并進行處理
func?consumer(ch?<-chan?Task)?{
?for?task?:=?range?ch?{
??fmt.Println("消費任務(wù):",?task.Data)
??//?執(zhí)行任務(wù)的處理邏輯
??time.Sleep(1?*?time.Second)?//?模擬處理任務(wù)的耗時
?}
}

func?main()?{
?//?創(chuàng)建一個任務(wù)通道,用于生產(chǎn)者和消費者之間的通信
?taskChannel?:=?make(chan?Task)

?//?啟動生產(chǎn)者Goroutine
?go?producer(taskChannel)

?//?啟動多個消費者Goroutine
?for?i?:=?1;?i?<=?3;?i++?{
??go?consumer(taskChannel)
?}

?//?阻塞主線程,保持程序運行
?select?{}
}

在上面的代碼種,使用了定時器(time.Timer)結(jié)合Goroutine來實現(xiàn)異步任務(wù)調(diào)度。通過啟動一個Goroutine來等待定時器的觸發(fā)事件,并執(zhí)行相應(yīng)的任務(wù)。這樣可以在后臺自動執(zhí)行任務(wù),而不需要阻塞主線程。

到此這篇關(guān)于4個場景教會你Go中Goroutine和通道是怎么用的的文章就介紹到這了,更多相關(guān)Go Goroutine通道內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 如何在Go中使用切片容量和長度

    如何在Go中使用切片容量和長度

    這篇文章主要介紹了如何在Go中使用切片容量和長度,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-11-11
  • golang實現(xiàn)java uuid的序列化方法

    golang實現(xiàn)java uuid的序列化方法

    這篇文章主要介紹了golang實現(xiàn)java uuid的序列化方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-09-09
  • CSP communicating sequential processes并發(fā)模型

    CSP communicating sequential processes并發(fā)模型

    這篇文章主要為大家介紹了CSP communicating sequential processes并發(fā)模型,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-05-05
  • Go語言異常處理error、panic、recover的使用

    Go語言異常處理error、panic、recover的使用

    GO語言中引入的異常的處理方式為error、panic、recover ,本文主要介紹了Go語言異常處理error、panic、recover的使用,感興趣的可以了解一下
    2024-08-08
  • Go?container包的介紹

    Go?container包的介紹

    這篇文章主要介紹了Go?container包,go語言container包中有List和Element容器ist和Element都是結(jié)構(gòu)體類型。結(jié)構(gòu)體類型有一個特點,那就是它們的零值都會是擁有其特定結(jié)構(gòu),但沒有任何定制化內(nèi)容的值,相當(dāng)于一個空殼,下面一起進文章來了解具體內(nèi)容吧
    2021-12-12
  • go語言實現(xiàn)銀行卡號Luhn校驗

    go語言實現(xiàn)銀行卡號Luhn校驗

    這篇文章主要為大家介紹了go語言Luhn校驗測試銀行卡號碼的示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-05-05
  • Go 語言中關(guān)于接口的三個

    Go 語言中關(guān)于接口的三個

    這篇文章主要介紹了Go 語言中關(guān)于接口的三個"潛規(guī)則",本文通過實例代碼相結(jié)合給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-06-06
  • Go Asynq異步任務(wù)處理的實現(xiàn)

    Go Asynq異步任務(wù)處理的實現(xiàn)

    Asynq是一個新興的異步任務(wù)處理解決方案,它提供了輕量級的、易于使用的API,本文主要介紹了Go Asynq異步任務(wù)處理的實現(xiàn),具有一定的參考價值,感興趣的可以了解一下
    2023-06-06
  • go?singleflight緩存雪崩源碼分析與應(yīng)用

    go?singleflight緩存雪崩源碼分析與應(yīng)用

    這篇文章主要為大家介紹了go?singleflight緩存雪崩源碼分析與應(yīng)用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-09-09
  • Golang語言如何高效拼接字符串詳解

    Golang語言如何高效拼接字符串詳解

    最近在做性能優(yōu)化,有個函數(shù)里面的耗時特別長,看里面的操作大多是一些字符串拼接的操作,而字符串拼接在 golang 里面其實有很多種實現(xiàn),下面這篇文章主要給大家介紹了關(guān)于Golang語言如何高效拼接字符串的相關(guān)資料,需要的朋友可以參考下
    2021-11-11

最新評論