婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > 深入Go goroutine理解

深入Go goroutine理解

熱門標簽:西部云谷一期地圖標注 高德地圖標注口訣 南通如皋申請開通400電話 江西轉化率高的羿智云外呼系統 地圖標注的汽車標 浙江高速公路地圖標注 中國地圖標注省會高清 廣州呼叫中心外呼系統 學海導航地圖標注

Go語言最大的特色就是從語言層面支持并發(Goroutine),Goroutine是Go中最基本的執行單元。事實上每一個Go程序至少有一個Goroutine:主Goroutine。當程序啟動時,它會自動創建。

為了更好理解Goroutine,現講一下線程和協程的概念

線程(Thread):有時被稱為輕量級進程(Lightweight Process,LWP),是程序執行流的最小單元。一個標準的線程由線程ID,當前指令指針(PC),寄存器集合和堆棧組成。另外,線程是進程中的一個實體,是被系統獨立調度和分派的基本單位,線程自己不擁有系統資源,只擁有一點兒在運行中必不可少的資源,但它可與同屬一個進程的其它線程共享進程所擁有的全部資源。

線程擁有自己獨立的棧和共享的堆,共享堆,不共享棧,線程的切換一般也由操作系統調度。

協程(coroutine):又稱微線程與子例程(或者稱為函數)一樣,協程(coroutine)也是一種程序組件。相對子例程而言,協程更為一般和靈活,但在實踐中使用沒有子例程那樣廣泛。

和線程類似,共享堆,不共享棧,協程的切換一般由程序員在代碼中顯式控制。它避免了上下文切換的額外耗費,兼顧了多線程的優點,簡化了高并發程序的復雜。

Goroutine和其他語言的協程(coroutine)在使用方式上類似,但從字面意義上來看不同(一個是Goroutine,一個是coroutine),再就是協程是一種協作任務控制機制,在最簡單的意義上,協程不是并發的,而Goroutine支持并發的。因此Goroutine可以理解為一種Go語言的協程。同時它可以運行在一個或多個線程上。

先給個簡單實例

func loop() {
  for i := 0; i  ; i++ {
    fmt.Printf("%d ", i)
  }
}

func main() {
  go loop() // 啟動一個goroutine
  loop()
}

GO并發的實現原理

一、Go并發模型

Go實現了兩種并發形式。第一種是大家普遍認知的:多線程共享內存。其實就是Java或者C++等語言中的多線程開發。另外一種是Go語言特有的,也是Go語言推薦的:CSP(communicating sequential processes)并發模型。

CSP并發模型是在1970年左右提出的概念,屬于比較新的概念,不同于傳統的多線程通過共享內存來通信,CSP講究的是“以通信的方式來共享內存”。

請記住下面這句話:
DO NOT COMMUNICATE BY SHARING MEMORY; INSTEAD, SHARE MEMORY BY COMMUNICATING.
“不要以共享內存的方式來通信,相反,要通過通信來共享內存。”

普通的線程并發模型,就是像Java、C++、或者Python,他們線程間通信都是通過共享內存的方式來進行的。非常典型的方式就是,在訪問共享數據(例如數組、Map、或者某個結構體或對象)的時候,通過鎖來訪問,因此,在很多時候,衍生出一種方便操作的數據結構,叫做“線程安全的數據結構”。例如Java提供的包”java.util.concurrent”中的數據結構。Go中也實現了傳統的線程并發模型。

Go的CSP并發模型,是通過goroutinechannel來實現的。

  • goroutine 是Go語言中并發的執行單位。有點抽象,其實就是和傳統概念上的”線程“類似,可以理解為”線程“
  • channel是Go語言中各個并發結構體(goroutine)之前的通信機制。 通俗的講,就是各個goroutine之間通信的”管道“,有點類似于Linux中的管道。

生成一個goroutine的方式非常的簡單:Go一下,就生成了。

go f();

通信機制channel也很方便,傳數據用channel - data,取數據用-channel

在通信過程中,傳數據channel - data和取數據-channel必然會成對出現,因為這邊傳,那邊取,兩個goroutine之間才會實現通信。

而且不管傳還是取,必阻塞,直到另外的goroutine傳或者取為止。

示例如下:

package main

import "fmt"

func main() {
  
  messages := make(chan string)

  go func() { messages - "ping" }()

  msg := -messages
  fmt.Println(msg)
}

注意 main()本身也是運行了一個goroutine。

messages:= make(chan int) 這樣就聲明了一個阻塞式的無緩沖的通道

chan 是關鍵字 代表我要創建一個通道

GO并發模型的實現原理

我們先從線程講起,無論語言層面何種并發模型,到了操作系統層面,一定是以線程的形態存在的。而操作系統根據資源訪問權限的不同,體系架構可分為用戶空間和內核空間;內核空間主要操作訪問CPU資源、I/O資源、內存資源等硬件資源,為上層應用程序提供最基本的基礎資源,用戶空間呢就是上層應用程序的固定活動空間,用戶空間不可以直接訪問資源,必須通過“系統調用”、“庫函數”或“Shell腳本”來調用內核空間提供的資源。

我們現在的計算機語言,可以狹義的認為是一種“軟件”,它們中所謂的“線程”,往往是用戶態的線程,和操作系統本身內核態的線程(簡稱KSE),還是有區別的。

線程模型的實現,可以分為以下幾種方式:

用戶級線程模型

如圖所示,多個用戶態的線程對應著一個內核線程,程序線程的創建、終止、切換或者同步等線程工作必須自身來完成。它可以做快速的上下文切換。缺點是不能有效利用多核CPU。

內核級線程模型

這種模型直接調用操作系統的內核線程,所有線程的創建、終止、切換、同步等操作,都由內核來完成。一個用戶態的線程對應一個系統線程,它可以利用多核機制,但上下文切換需要消耗額外的資源。C++就是這種。

兩級線程模型

這種模型是介于用戶級線程模型和內核級線程模型之間的一種線程模型。這種模型的實現非常復雜,和內核級線程模型類似,一個進程中可以對應多個內核級線程,但是進程中的線程不和內核線程一一對應;這種線程模型會先創建多個內核級線程,然后用自身的用戶級線程去對應創建的多個內核級線程,自身的用戶級線程需要本身程序去調度,內核級的線程交給操作系統內核去調度。

M個用戶線程對應N個系統線程,缺點增加了調度器的實現難度。

Go語言的線程模型就是一種特殊的兩級線程模型(GPM調度模型)。

Go線程實現模型MPG

M指的是Machine,一個M直接關聯了一個內核線程。由操作系統管理。
P指的是”processor”,代表了M所需的上下文環境,也是處理用戶級代碼邏輯的處理器。它負責銜接M和G的調度上下文,將等待執行的G與M對接。
G指的是Goroutine,其實本質上也是一種輕量級的線程。包括了調用棧,重要的調度信息,例如channel等。

P的數量由環境變量中的GOMAXPROCS決定,通常來說它是和核心數對應,例如在4Core的服務器上回啟動4個線程。G會有很多個,每個P會將Goroutine從一個就緒的隊列中做Pop操作,為了減小鎖的競爭,通常情況下每個P會負責一個隊列。

三者關系如下圖所示:

以上這個圖講的是兩個線程(內核線程)的情況。一個M會對應一個內核線程,一個M也會連接一個上下文P,一個上下文P相當于一個“處理器”,一個上下文連接一個或者多個Goroutine。為了運行goroutine,線程必須保存上下文。

上下文P(Processor)的數量在啟動時設置為GOMAXPROCS環境變量的值或通過運行時函數GOMAXPROCS()。通常情況下,在程序執行期間不會更改。上下文數量固定意味著只有固定數量的線程在任何時候運行Go代碼。我們可以使用它來調整Go進程到個人計算機的調用,例如4核PC在4個線程上運行Go代碼。

圖中P正在執行的Goroutine為藍色的;處于待執行狀態的Goroutine為灰色的,灰色的Goroutine形成了一個隊列runqueues

Go語言里,啟動一個goroutine很容易:go function 就行,所以每有一個go語句被執行,runqueue隊列就在其末尾加入一個goroutine,一旦上下文運行goroutine直到調度點,它會從其runqueue中彈出goroutine,設置堆棧和指令指針并開始運行goroutine。

拋棄P(Processor)

你可能會想,為什么一定需要一個上下文,我們能不能直接除去上下文,讓Goroutinerunqueues掛到M上呢?答案是不行,需要上下文的目的,是讓我們可以直接放開其他線程,當遇到內核線程阻塞的時候。

一個很簡單的例子就是系統調用sysall,一個線程肯定不能同時執行代碼和系統調用被阻塞,這個時候,此線程M需要放棄當前的上下文環境P,以便可以讓其他的Goroutine被調度執行。

如上圖左圖所示,M0中的G0執行了syscall,然后就創建了一個M1(也有可能來自線程緩存),(轉向右圖)然后M0丟棄了P,等待syscall的返回值,M1接受了P,將·繼續執行Goroutine隊列中的其他Goroutine

當系統調用syscall結束后,M0會“偷”一個上下文,如果不成功,M0就把它的Gouroutine G0放到一個全局的runqueue中,將自己置于線程緩存中并進入休眠狀態。全局runqueue是各個P在運行完自己的本地的Goroutine runqueue后用來拉取新goroutine的地方。P也會周期性的檢查這個全局runqueue上的goroutine,否則,全局runqueue上的goroutines可能得不到執行而餓死。

均衡的分配工作

按照以上的說法,上下文P會定期的檢查全局的goroutine 隊列中的goroutine,以便自己在消費掉自身Goroutine隊列的時候有事可做。假如全局goroutine隊列中的goroutine也沒了呢?就從其他運行的中的P的runqueue里偷。

每個P中的Goroutine不同導致他們運行的效率和時間也不同,在一個有很多P和M的環境中,不能讓一個P跑完自身的Goroutine就沒事可做了,因為或許其他的P有很長的goroutine隊列要跑,得需要均衡。
該如何解決呢?

Go的做法倒也直接,從其他P中偷一半!

Goroutine 小結

優點:

1、開銷小

POSIX的thread API雖然能夠提供豐富的API,例如配置自己的CPU親和性,申請資源等等,線程在得到了很多與進程相同的控制權的同時,開銷也非常的大,在Goroutine中則不需這些額外的開銷,所以一個Golang的程序中可以支持10w級別的Goroutine。

每個 goroutine (協程) 默認占用內存遠比 Java 、C 的線程少(goroutine:2KB ,線程:8MB)

2、調度性能好

在Golang的程序中,操作系統級別的線程調度,通常不會做出合適的調度決策。例如在GC時,內存必須要達到一個一致的狀態。在Goroutine機制里,Golang可以控制Goroutine的調度,從而在一個合適的時間進行GC。

在應用層模擬的線程,它避免了上下文切換的額外耗費,兼顧了多線程的優點。簡化了高并發程序的復雜度。

缺點:

協程調度機制無法實現公平調度。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

您可能感興趣的文章:
  • GOLANG使用Context管理關聯goroutine的方法
  • Golang 探索對Goroutine的控制方法(詳解)
  • 關于Golang中for-loop與goroutine的問題詳解
  • go語言執行等待直到后臺goroutine執行完成實例分析
  • Go語言輕量級線程Goroutine用法實例

標簽:保定 常州 德宏 許昌 東營 曲靖 吐魯番 貴州

巨人網絡通訊聲明:本文標題《深入Go goroutine理解》,本文關鍵詞  深入,goroutine,理解,深入,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《深入Go goroutine理解》相關的同類信息!
  • 本頁收集關于深入Go goroutine理解的相關信息資訊供網民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    www国产亚洲精品久久麻豆| 欧美三级在线播放| 色综合一区二区三区| 欧美成人性福生活免费看| 日日夜夜精品视频天天综合网| 91同城在线观看| 中文字幕亚洲视频| 91免费国产在线| 一区av在线播放| 欧美日韩精品一区二区三区四区 | 国产酒店精品激情| 国产日韩欧美高清在线| av电影天堂一区二区在线观看| 国产精品乱人伦一区二区| av亚洲产国偷v产偷v自拍| 亚洲激情成人在线| 91精品在线免费观看| 国产乱一区二区| 亚洲欧美一区二区久久| 337p亚洲精品色噜噜狠狠| 国产一区二区在线影院| **网站欧美大片在线观看| 欧美影院精品一区| 精品一区二区三区视频| 亚洲人成人一区二区在线观看 | 美女网站色91| 国产欧美一区视频| 一本大道久久a久久综合婷婷 | 一区二区不卡在线播放| 日韩欧美中文字幕公布| 成人av在线看| 丝袜a∨在线一区二区三区不卡| 久久久精品免费免费| 在线国产亚洲欧美| 国产成人免费视频| 日本中文字幕一区二区有限公司| 国产亚洲欧美中文| 在线播放日韩导航| 成人激情免费电影网址| 美国毛片一区二区三区| 亚洲与欧洲av电影| 国产精品久久综合| 欧美大片日本大片免费观看| 在线免费观看不卡av| 成人午夜av电影| 精品影院一区二区久久久| 亚洲综合视频在线| 亚洲欧洲日产国产综合网| 亚洲精品在线一区二区| 91精品欧美福利在线观看| 色天天综合久久久久综合片| 国产精品中文字幕日韩精品 | 欧美色综合网站| www.av精品| 国产91在线观看丝袜| 国产在线精品一区在线观看麻豆| 久久精品亚洲国产奇米99| 欧美色图在线观看| 北条麻妃国产九九精品视频| 国产一区二三区| 美国一区二区三区在线播放| 午夜精品久久久久| 亚洲国产中文字幕| 亚洲国产日韩a在线播放性色| 亚洲男人的天堂av| 亚洲黄色片在线观看| 亚洲精品国产品国语在线app| 久久综合丝袜日本网| 91黄色免费版| 91视频精品在这里| 色婷婷精品久久二区二区蜜臀av| 一本久久综合亚洲鲁鲁五月天| 99久久777色| 黄一区二区三区| 国产91丝袜在线18| 91片在线免费观看| 色婷婷综合久色| 欧美少妇xxx| 欧美私模裸体表演在线观看| 欧美日韩在线一区二区| 91丨九色丨尤物| 欧美日韩的一区二区| 91精品国产欧美一区二区成人| 日韩欧美美女一区二区三区| 国产调教视频一区| 精品日韩一区二区| 亚洲欧洲精品一区二区三区不卡| 亚洲人xxxx| 日本亚洲视频在线| 国产成人午夜精品影院观看视频| 99在线热播精品免费| 欧美日产国产精品| 久久综合一区二区| 亚洲精品中文字幕乱码三区| 日韩电影免费在线观看网站| 国产一区在线不卡| 91网站最新地址| 日韩欧美国产成人一区二区| 国产精品久久久久久久岛一牛影视| 亚洲午夜一二三区视频| 国产在线精品国自产拍免费| 欧美亚洲精品一区| 欧美国产乱子伦| 日韩高清不卡在线| 99视频国产精品| 精品动漫一区二区三区在线观看| 亚洲人精品午夜| 国产成人在线视频网址| 欧美精品久久99久久在免费线| 欧美国产精品v| 日韩av午夜在线观看| 波多野结衣在线aⅴ中文字幕不卡| 8x8x8国产精品| 中文字幕一区二区三区av| 青娱乐精品在线视频| 色婷婷狠狠综合| 国产女同互慰高潮91漫画| 青青草视频一区| 欧美午夜精品久久久久久孕妇 | 99国产精品国产精品毛片| 日韩美女一区二区三区| 亚洲精品国产无套在线观| 国产美女视频一区| 日韩欧美一区在线观看| 亚洲福利一区二区三区| 国产成人av电影在线| 精品少妇一区二区三区在线视频 | 一区二区高清免费观看影视大全| 国内不卡的二区三区中文字幕| 91精品国产一区二区三区蜜臀| 亚洲视频在线观看三级| 国产精品一区二区免费不卡 | 天天影视涩香欲综合网| 色综合久久综合| 中文字幕中文字幕中文字幕亚洲无线| 久久97超碰国产精品超碰| 日韩一卡二卡三卡四卡| 日日欢夜夜爽一区| 欧美精品高清视频| 亚洲444eee在线观看| 欧美丝袜第三区| 亚洲成人www| 91精品国产欧美一区二区18| 免费观看成人av| 日韩精品中文字幕在线不卡尤物 | 国产欧美精品一区二区色综合 | 色八戒一区二区三区| 又紧又大又爽精品一区二区| 在线欧美日韩精品| 性做久久久久久免费观看| 欧美精品久久久久久久多人混战 | 亚洲欧美偷拍卡通变态| 在线看国产一区二区| 五月婷婷综合在线| 欧美大胆一级视频| 成人一级片网址| 亚洲最快最全在线视频| 欧美裸体bbwbbwbbw| 美脚の诱脚舐め脚责91| 久久久蜜臀国产一区二区| av在线不卡网| 午夜视频在线观看一区| 精品美女在线观看| 91一区一区三区| 天堂资源在线中文精品| 久久精品亚洲乱码伦伦中文 | 国产精品久久久99| 在线视频综合导航| 国内欧美视频一区二区 | 国产高清成人在线| 一区二区三区欧美视频| 欧美一个色资源| 成人av电影在线网| 日本成人在线不卡视频| 国产精品色噜噜| 欧美老年两性高潮| 高清不卡一区二区| 国产91高潮流白浆在线麻豆| 亚洲欧洲精品一区二区三区| 中文字幕乱码一区二区免费| 亚洲综合一区在线| 亚洲在线视频网站| 成人激情综合网站| 国产亚洲女人久久久久毛片| 欧美日韩国产小视频| 91国偷自产一区二区使用方法| 亚洲精品v日韩精品| 欧美日韩国产精品自在自线| 亚洲视频网在线直播| 日韩av中文在线观看| 亚洲精品一区二区三区蜜桃下载 | 中文字幕在线免费不卡| 欧美午夜在线观看| 在线观看日韩电影| 亚洲国产精华液网站w| 毛片基地黄久久久久久天堂| 美洲天堂一区二卡三卡四卡视频| 欧美精品一区二区久久婷婷| 91精品国产黑色紧身裤美女| 在线视频观看一区|