淺析Golang中的協(xié)程(goroutine)
進(jìn)程
進(jìn)程就是程序在操作系統(tǒng)中的一次執(zhí)行過(guò)程,是系統(tǒng)進(jìn)行資源分配和調(diào)度的基本單位,進(jìn)程是一個(gè)動(dòng)態(tài)概念,是程序在執(zhí)行過(guò)程中分配和管理資源的基本單位,每一個(gè)進(jìn)程都有一個(gè)自己的地址空間。一個(gè)進(jìn)程至少有5種基本狀態(tài):初始狀態(tài),執(zhí)行狀態(tài),等待狀態(tài),就緒狀態(tài),終止?fàn)顟B(tài)。通俗的講,進(jìn)程就是一個(gè)正在執(zhí)行的程序。
線程
線程是進(jìn)程的一個(gè)執(zhí)行實(shí)例,是程序執(zhí)行的最小單元,它是比進(jìn)程更小的能獨(dú)立運(yùn)行的基本單位。一個(gè)進(jìn)程可以創(chuàng)建多個(gè)線程,同一個(gè)進(jìn)程中的多個(gè)線程可以并發(fā)執(zhí)行,一個(gè)程序要運(yùn)行的話至少有一個(gè)進(jìn)程。
并發(fā)
并發(fā)是指在同一時(shí)間段內(nèi)處理多個(gè)任務(wù),通過(guò)多個(gè)任務(wù)之間的切換,使得在表面上看來(lái)是同時(shí)進(jìn)行的。在 Go 語(yǔ)言中,可以使用 goroutine 和 channel 實(shí)現(xiàn)并發(fā)編程。
特點(diǎn):
多個(gè)任務(wù)作用在一個(gè)CPU上面
同一時(shí)間點(diǎn)只能有一個(gè)任務(wù)執(zhí)行
同一時(shí)間段內(nèi)執(zhí)行多個(gè)任務(wù)
并行
并行是指同時(shí)處理多個(gè)任務(wù),即多個(gè)任務(wù)同時(shí)在不同的處理器上進(jìn)行執(zhí)行。并行可以顯著提高程序的性能,特別是在多核 CPU 中,能夠利用多個(gè) CPU 核心進(jìn)行計(jì)算。在 Go 語(yǔ)言中,可以使用 goroutine 和 runtime 包的 GOMAXPROCS 函數(shù)實(shí)現(xiàn)并行編程。
特點(diǎn):
多個(gè)任務(wù)作用在多個(gè)CPU上面
同一時(shí)刻執(zhí)行多個(gè)任務(wù)
通俗的講,多線程程序在單核CPU上面運(yùn)行就是并發(fā),多線程程序在多核CUP上運(yùn)行就是并行,如果線程數(shù)大于CPU核數(shù),則多線程程序在多個(gè)CPU上面運(yùn)行既有并行也有并發(fā)。
協(xié)程(goroutine)
在 Go 語(yǔ)言中,協(xié)程(goroutine)是輕量級(jí)的線程,它是 Go 語(yǔ)言中實(shí)現(xiàn)并發(fā)編程的基礎(chǔ)。與傳統(tǒng)的線程相比,協(xié)程的創(chuàng)建和切換都非常輕量級(jí),可以在單個(gè)線程內(nèi)創(chuàng)建成千上萬(wàn)個(gè)協(xié)程,且切換開(kāi)銷非常小,因此可以實(shí)現(xiàn)高效的并發(fā)編程。
Go 語(yǔ)言中的協(xié)程是由 Go 運(yùn)行時(shí)調(diào)度器(scheduler)進(jìn)行管理和調(diào)度的。當(dāng)程序啟動(dòng)時(shí),Go 運(yùn)行時(shí)會(huì)默認(rèn)啟動(dòng)一個(gè)主協(xié)程,主協(xié)程會(huì)創(chuàng)建其他的子協(xié)程,這些協(xié)程會(huì)被分配到不同的系統(tǒng)線程上進(jìn)行執(zhí)行。當(dāng)某個(gè)協(xié)程發(fā)生阻塞時(shí),Go 運(yùn)行時(shí)會(huì)將該協(xié)程掛起并讓出 CPU,轉(zhuǎn)而執(zhí)行其他協(xié)程,以充分利用系統(tǒng)資源。
在 Go 語(yǔ)言中,創(chuàng)建協(xié)程非常簡(jiǎn)單,只需要在函數(shù)調(diào)用前加上 go 關(guān)鍵字即可。
例如:
func main() { go func() { fmt.Println("Hello, world!") }() }
使用sync.WaitGroup等待協(xié)程執(zhí)行完畢
func test1() { for i := 0; i < 10; i++ { //每100毫秒輸出一次 fmt.Println("test1() 你好Golang-", i) time.Sleep(time.Millisecond * 100) } wg.Done() //協(xié)程計(jì)數(shù)器加-1 } func main() { //注意: //1.主線程執(zhí)行完畢后即使協(xié)程沒(méi)有執(zhí)行完畢程序也會(huì)退出 //2.協(xié)程可以在主線程沒(méi)有執(zhí)行完畢前提前退出,協(xié)程是否執(zhí)行完畢不會(huì)影響主線程的執(zhí)行 //為了保證我們的程序可以順利執(zhí)行,我們想讓協(xié)程執(zhí)行完畢后再執(zhí)行主進(jìn)程退出, //這個(gè)時(shí)候我們可以使用sync.WaitGroup等待協(xié)程執(zhí)行完畢 wg.Add(1) //協(xié)程計(jì)數(shù)器加1 go test1() //表示開(kāi)啟一個(gè)協(xié)程 for i := 0; i < 10; i++ { //每50毫秒輸出一次 fmt.Println("mian() 你好Golang-", i) time.Sleep(time.Millisecond * 50) } wg.Wait() //等待協(xié)程執(zhí)行完畢 fmt.Println("主線程退出。。。") }
多協(xié)程和多線程
Golang中每個(gè)goroutine(協(xié)程)默認(rèn)占用內(nèi)存比Java、C的線程少。OS線程(操作系統(tǒng)線程)一般都有固定的棧內(nèi)存(通常為2MB左右),一個(gè)goroutine(協(xié)程)占用內(nèi)存非常少,只有2KB左右,多協(xié)程切換調(diào)度開(kāi)銷方面遠(yuǎn)比線程要少。這也是為什么越來(lái)越多的大公司使用Golang的原因之一。
func test1() { for i := 0; i < 10; i++ { //每100毫秒輸出一次 fmt.Println("test1() 你好Golang-", i) time.Sleep(time.Millisecond * 100) } wg.Done() //協(xié)程計(jì)數(shù)器加-1 } func test2() { for i := 0; i < 10; i++ { //每100毫秒輸出一次 fmt.Println("test2() 你好Golang-", i) time.Sleep(time.Millisecond * 100) } wg.Done() //協(xié)程計(jì)數(shù)器加-1 } func main() { wg.Add(1) //協(xié)程計(jì)數(shù)器加1 go test1() //表示開(kāi)啟一個(gè)協(xié)程 wg.Add(1) //協(xié)程計(jì)數(shù)器加1 go test2() //表示開(kāi)啟一個(gè)協(xié)程 wg.Wait() fmt.Println("主程序退出。。。") }
以上就是淺析Golang中的協(xié)程(goroutine)的詳細(xì)內(nèi)容,更多關(guān)于Golang 協(xié)程的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
go語(yǔ)言使用第三方包 json化結(jié)構(gòu)體操作示例
這篇文章主要介紹了go語(yǔ)言使用第三方包 json化結(jié)構(gòu)體操作,結(jié)合實(shí)例形式分析了Go語(yǔ)言ffjson包git安裝及結(jié)構(gòu)體轉(zhuǎn)json字符串相關(guān)操作技巧,需要的朋友可以參考下2019-06-06詳解Go語(yǔ)言RESTful JSON API創(chuàng)建
這篇文章主要介紹了詳解Go語(yǔ)言RESTful JSON API創(chuàng)建,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-05-05Golang中HTTP路由設(shè)計(jì)的使用與實(shí)現(xiàn)
這篇文章主要介紹了Golang中HTTP路由設(shè)計(jì)的使用與實(shí)現(xiàn),為什么要設(shè)計(jì)路由規(guī)則,因?yàn)槁酚梢?guī)則是HTTP的請(qǐng)求按照一定的規(guī)則 ,匹配查找到對(duì)應(yīng)的控制器并傳遞執(zhí)行的邏輯,需要的朋友可以參考下2023-05-05go語(yǔ)言計(jì)算兩個(gè)時(shí)間的時(shí)間差方法
這篇文章主要介紹了go語(yǔ)言計(jì)算兩個(gè)時(shí)間的時(shí)間差方法,涉及Python操作時(shí)間的技巧,需要的朋友可以參考下2015-03-03