女人自慰AV免费观看内涵网,日韩国产剧情在线观看网址,神马电影网特片网,最新一级电影欧美,在线观看亚洲欧美日韩,黄色视频在线播放免费观看,ABO涨奶期羡澄,第一导航fulione,美女主播操b

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Go并發(fā)模型的實(shí)現(xiàn)原理

馬哥Linux運(yùn)維 ? 來源:CSDN技術(shù)社區(qū) ? 作者:CSDN技術(shù)社區(qū) ? 2022-04-15 08:49 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

Go語言是為并發(fā)而生的語言,Go語言是為數(shù)不多的在語言層面實(shí)現(xiàn)并發(fā)的語言;也正是Go語言的并發(fā)特性,吸引了全球無數(shù)的開發(fā)者

并發(fā)(concurrency)和并行(parallellism)

  • 并發(fā)(concurrency):兩個(gè)或兩個(gè)以上的任務(wù)在一段時(shí)間內(nèi)被執(zhí)行。我們不必care這些任務(wù)在某一個(gè)時(shí)間點(diǎn)是否是同時(shí)執(zhí)行,可能同時(shí)執(zhí)行,也可能不是,我們只關(guān)心在一段時(shí)間內(nèi),哪怕是很短的時(shí)間(一秒或者兩秒)是否執(zhí)行解決了兩個(gè)或兩個(gè)以上任務(wù)。

  • 并行(parallellism):兩個(gè)或兩個(gè)以上的任務(wù)在同一時(shí)刻被同時(shí)執(zhí)行。

并發(fā)說的是邏輯上的概念,而并行,強(qiáng)調(diào)的是物理運(yùn)行狀態(tài)。并發(fā)“包含”并行。

(詳情請(qǐng)見:Rob Pike 的PPT

Go的CSP并發(fā)模型

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

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

請(qǐng)記住下面這句話:"Do not communicate by sharing memory; instead, share memory by communicating."“不要以共享內(nèi)存的方式來通信,相反,要通過通信來共享內(nèi)存。”

普通的線程并發(fā)模型,就是像Java、C++、或者Python,他們線程間通信都是通過共享內(nèi)存的方式來進(jìn)行的。非常典型的方式就是,在訪問共享數(shù)據(jù)(例如數(shù)組、Map、或者某個(gè)結(jié)構(gòu)體或?qū)ο螅┑臅r(shí)候,通過鎖來訪問,因此,在很多時(shí)候,衍生出一種方便操作的數(shù)據(jù)結(jié)構(gòu),叫做“線程安全的數(shù)據(jù)結(jié)構(gòu)”。例如Java提供的包”java.util.concurrent”中的數(shù)據(jù)結(jié)構(gòu)。Go中也實(shí)現(xiàn)了傳統(tǒng)的線程并發(fā)模型。

Go的CSP并發(fā)模型,是通過goroutinechannel來實(shí)現(xiàn)的。

  • goroutine 是Go語言中并發(fā)的執(zhí)行單位。有點(diǎn)抽象,其實(shí)就是和傳統(tǒng)概念上的”線程“類似,可以理解為”線程“。

  • channel 是Go語言中各個(gè)并發(fā)結(jié)構(gòu)體(goroutine)之前的通信機(jī)制。通俗的講,就是各個(gè)goroutine之間通信的”管道“,有點(diǎn)類似于Linux中的管道。

生成一個(gè)goroutine的方式非常的簡(jiǎn)單:Go一下,就生成了:


	

gof();

通信機(jī)制channel也很方便,傳數(shù)據(jù)用channel <- data,取數(shù)據(jù)用<-channel。

在通信過程中,傳數(shù)據(jù)channel <- data和取數(shù)據(jù)<-channel必然會(huì)成對(duì)出現(xiàn),因?yàn)檫@邊傳,那邊取,兩個(gè)goroutine之間才會(huì)實(shí)現(xiàn)通信。

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

有兩個(gè)goroutine,其中一個(gè)發(fā)起了向channel中發(fā)起了傳值操作。(goroutine為矩形,channel為箭頭)

8287b7c8-bc16-11ec-aa7f-dac502259ad0.png

左邊的goroutine開始阻塞,等待有人接收。

這時(shí)候,右邊的goroutine發(fā)起了接收操作

82943c14-bc16-11ec-aa7f-dac502259ad0.png

右邊的goroutine也開始阻塞,等待別人傳送。

這時(shí)候,兩邊goroutine都發(fā)現(xiàn)了對(duì)方,于是兩個(gè)goroutine開始一傳,一收。

82a08a0a-bc16-11ec-aa7f-dac502259ad0.png

這便是Golang CSP并發(fā)模型最基本的形式。

Go并發(fā)模型的實(shí)現(xiàn)原理

我們先從線程講起,無論語言層面何種并發(fā)模型,到了操作系統(tǒng)層面,一定是以線程的形態(tài)存在的。而操作系統(tǒng)根據(jù)資源訪問權(quán)限的不同,體系架構(gòu)可分為用戶空間和內(nèi)核空間;內(nèi)核空間主要操作訪問CPU資源、I/O資源、內(nèi)存資源等硬件資源,為上層應(yīng)用程序提供最基本的基礎(chǔ)資源,用戶空間呢就是上層應(yīng)用程序的固定活動(dòng)空間,用戶空間不可以直接訪問資源,必須通過“系統(tǒng)調(diào)用”、“庫函數(shù)”或“Shell腳本”來調(diào)用內(nèi)核空間提供的資源。

我們現(xiàn)在的計(jì)算機(jī)語言,可以狹義的認(rèn)為是一種“軟件”,它們中所謂的“線程”,往往是用戶態(tài)的線程,和操作系統(tǒng)本身內(nèi)核態(tài)的線程(簡(jiǎn)稱KSE),還是有區(qū)別的。

線程模型的實(shí)現(xiàn),可以分為以下幾種方式:

用戶級(jí)線程模型

如圖所示,多個(gè)用戶態(tài)的線程對(duì)應(yīng)著一個(gè)內(nèi)核線程,程序線程的創(chuàng)建、終止、切換或者同步等線程工作必須自身來完成。

82ac575e-bc16-11ec-aa7f-dac502259ad0.png

內(nèi)核級(jí)線程模型

這種模型直接調(diào)用操作系統(tǒng)的內(nèi)核線程,所有線程的創(chuàng)建、終止、切換、同步等操作,都由內(nèi)核來完成。C++就是這種。

82b7d8a4-bc16-11ec-aa7f-dac502259ad0.png

兩級(jí)線程模型

這種模型是介于用戶級(jí)線程模型和內(nèi)核級(jí)線程模型之間的一種線程模型。這種模型的實(shí)現(xiàn)非常復(fù)雜,和內(nèi)核級(jí)線程模型類似,一個(gè)進(jìn)程中可以對(duì)應(yīng)多個(gè)內(nèi)核級(jí)線程,但是進(jìn)程中的線程不和內(nèi)核線程一一對(duì)應(yīng);這種線程模型會(huì)先創(chuàng)建多個(gè)內(nèi)核級(jí)線程,然后用自身的用戶級(jí)線程去對(duì)應(yīng)創(chuàng)建的多個(gè)內(nèi)核級(jí)線程,自身的用戶級(jí)線程需要本身程序去調(diào)度,內(nèi)核級(jí)的線程交給操作系統(tǒng)內(nèi)核去調(diào)度。

Go語言的線程模型就是一種特殊的兩級(jí)線程模型。暫且叫它“MPG”模型吧。

82b7d8a4-bc16-11ec-aa7f-dac502259ad0.png

Go線程實(shí)現(xiàn)模型MPG

M指的是Machine,一個(gè)M直接關(guān)聯(lián)了一個(gè)內(nèi)核線程。P指的是”processor”,代表了M所需的上下文環(huán)境,也是處理用戶級(jí)代碼邏輯的處理器G指的是Goroutine,其實(shí)本質(zhì)上也是一種輕量級(jí)的線程。

三者關(guān)系如下圖所示:

82d3ec24-bc16-11ec-aa7f-dac502259ad0.png

以上這個(gè)圖講的是兩個(gè)線程(內(nèi)核線程)的情況。一個(gè)M會(huì)對(duì)應(yīng)一個(gè)內(nèi)核線程,一個(gè)M也會(huì)連接一個(gè)上下文P,一個(gè)上下文P相當(dāng)于一個(gè)“處理器”,一個(gè)上下文連接一個(gè)或者多個(gè)Goroutine。P(Processor)的數(shù)量是在啟動(dòng)時(shí)被設(shè)置為環(huán)境變量GOMAXPROCS的值,或者通過運(yùn)行時(shí)調(diào)用函數(shù)runtime.GOMAXPROCS()進(jìn)行設(shè)置。Processor數(shù)量固定意味著任意時(shí)刻只有固定數(shù)量的線程在運(yùn)行g(shù)o代碼。Goroutine中就是我們要執(zhí)行并發(fā)的代碼。圖中P正在執(zhí)行的Goroutine為藍(lán)色的;處于待執(zhí)行狀態(tài)的Goroutine為灰色的,灰色的Goroutine形成了一個(gè)隊(duì)列runqueues。

三者關(guān)系的宏觀的圖為:

82e01170-bc16-11ec-aa7f-dac502259ad0.png

拋棄 P(Processor)

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

一個(gè)很簡(jiǎn)單的例子就是系統(tǒng)調(diào)用sysall,一個(gè)線程肯定不能同時(shí)執(zhí)行代碼和系統(tǒng)調(diào)用被阻塞,這個(gè)時(shí)候,此線程M需要放棄當(dāng)前的上下文環(huán)境P,以便可以讓其他的Goroutine被調(diào)度執(zhí)行。

82eba81e-bc16-11ec-aa7f-dac502259ad0.png

如上圖左圖所示,M0中的G0執(zhí)行了syscall,然后就創(chuàng)建了一個(gè)M1(也有可能本身就存在,沒創(chuàng)建),(轉(zhuǎn)向右圖)然后M0丟棄了P,等待syscall的返回值,M1接受了P,將·繼續(xù)執(zhí)行Goroutine隊(duì)列中的其他Goroutine

當(dāng)系統(tǒng)調(diào)用syscall結(jié)束后,M0會(huì)“偷”一個(gè)上下文,如果不成功,M0就把它的Gouroutine G0放到一個(gè)全局的runqueue中,然后自己放到線程池或者轉(zhuǎn)入休眠狀態(tài)。全局runqueue是各個(gè)P在運(yùn)行完自己的本地的Goroutine runqueue后用來拉取新goroutine的地方。P也會(huì)周期性的檢查這個(gè)全局runqueue上的goroutine,否則,全局runqueue上的goroutines可能得不到執(zhí)行而餓死。

均衡的分配工作

按照以上的說法,上下文P會(huì)定期的檢查全局的goroutine 隊(duì)列中的goroutine,以便自己在消費(fèi)掉自身Goroutine隊(duì)列的時(shí)候有事可做。假如全局goroutine隊(duì)列中的goroutine也沒了呢?就從其他運(yùn)行的中的Prunqueue里偷。

每個(gè)P中的Goroutine不同導(dǎo)致他們運(yùn)行的效率和時(shí)間也不同,在一個(gè)有很多P和M的環(huán)境中,不能讓一個(gè)P跑完自身的Goroutine就沒事可做了,因?yàn)榛蛟S其他的P有很長(zhǎng)的goroutine隊(duì)列要跑,得需要均衡。該如何解決呢?

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

82f7fd94-bc16-11ec-aa7f-dac502259ad0.png

原文標(biāo)題:Golang 并發(fā)原理分析

文章出處:【微信公眾號(hào):馬哥Linux運(yùn)維】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

審核編輯:湯梓紅
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 原理
    +關(guān)注

    關(guān)注

    4

    文章

    550

    瀏覽量

    45325
  • 模型
    +關(guān)注

    關(guān)注

    1

    文章

    3504

    瀏覽量

    50206
  • go語言
    +關(guān)注

    關(guān)注

    1

    文章

    159

    瀏覽量

    9331

原文標(biāo)題:Golang 并發(fā)原理分析

文章出處:【微信號(hào):magedu-Linux,微信公眾號(hào):馬哥Linux運(yùn)維】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    三十分鐘入門基礎(chǔ)Go Java小子版

    語法與 C 相近,但功能上有:內(nèi)存安全,GC,結(jié)構(gòu)形態(tài)及 CSP-style 并發(fā)計(jì)算。 適用范圍 本篇文章適用于學(xué)習(xí)過其他面向?qū)ο笳Z言(Java、Php),但沒有學(xué)過Go語言的初學(xué)者。文章 主要從Go
    的頭像 發(fā)表于 08-12 14:32 ?958次閱讀
    三十分鐘入門基礎(chǔ)<b class='flag-5'>Go</b> Java小子版

    【GoRK3288】1.Rockchip RK3288, GO!GO!!GO!!!

    技術(shù)的興起,使用Go的高并發(fā)特性來實(shí)現(xiàn)控制服務(wù)器將會(huì)大大提高運(yùn)行的性能簡(jiǎn)化開發(fā)的難度,而且可以作為一個(gè)節(jié)點(diǎn)控制著各種外設(shè)。這個(gè)節(jié)點(diǎn)設(shè)備甚至可以是ARM系統(tǒng),并且核心越多性能越高,所以本次開發(fā)采用了高性能
    發(fā)表于 08-14 21:07

    Go語言開發(fā)有什么優(yōu)勢(shì)?怎么學(xué)?

    、庫的依賴關(guān)系,大大減輕了維護(hù)的負(fù)擔(dān)。  2. 并發(fā)行好。Go天生為高并發(fā)而生,Goroutine 和 channel 使得編寫高并發(fā)的服務(wù)端軟件變得相當(dāng)容易,很多情況下完全不需要考慮
    發(fā)表于 12-19 16:08

    Go開發(fā)語言的優(yōu)勢(shì)在哪里?

    Goroutine和 channel 使得編寫高并發(fā)的服務(wù)端軟件變得相當(dāng)容易,很多情況下完全不需要考慮鎖機(jī)制以及由此帶來的各種問題。單個(gè) Go 應(yīng)用也能有效的利用多個(gè) CPU 核,并行執(zhí)行的性能好。3.
    發(fā)表于 03-22 15:04

    Lite Actor:方舟Actor并發(fā)模型的輕量級(jí)優(yōu)化

    并發(fā)模型是用來實(shí)現(xiàn)不同應(yīng)用場(chǎng)景中并發(fā)任務(wù)的編程模型,通過合理地使用多線程,可以縮減應(yīng)用程序的開發(fā)和維護(hù)成本,同時(shí)還能更好地提升應(yīng)用程序在多核
    發(fā)表于 07-18 12:00

    Go語言及Beego框架環(huán)境搭建相關(guān)資料推薦

    就完成了。  天生支持并發(fā),可以充分的利用多核,很容易實(shí)現(xiàn)并發(fā)。  25個(gè)關(guān)鍵字,但是表達(dá)能力很強(qiáng)大,幾乎支持大多數(shù)你在其他語言見過的特性:繼承、重載、對(duì)象等?! ?nèi)置強(qiáng)大的工具,Go
    發(fā)表于 10-17 16:57

    移動(dòng)應(yīng)用高級(jí)語言開發(fā)——并發(fā)探索

    、簡(jiǎn)單、高效的并發(fā)模型考慮,業(yè)界目前給應(yīng)用開發(fā)者提供的多線程模型,有避免數(shù)據(jù)競(jìng)爭(zhēng)、實(shí)現(xiàn)無鎖化的趨勢(shì)。例如,在Web上,JS在多線程使用的是消息通信機(jī)制(內(nèi)存隔離,增加可轉(zhuǎn)移Builti
    發(fā)表于 08-28 17:08

    負(fù)荷管理系統(tǒng)中的并發(fā)通信設(shè)計(jì)與實(shí)現(xiàn)

    負(fù)荷管理系統(tǒng)中的并發(fā)通信設(shè)計(jì)與實(shí)現(xiàn)摘 要 大規(guī)模并發(fā)通信的管理與控制是計(jì)算機(jī)監(jiān)控領(lǐng)域研究的熱點(diǎn)與難點(diǎn)之一。本文以廈門電業(yè)局負(fù)荷管理系統(tǒng)為例,對(duì)大規(guī)模并發(fā)通信
    發(fā)表于 11-01 09:50 ?13次下載

    dubbo-go 中的 TPS Limit 設(shè)計(jì)與實(shí)現(xiàn)

    則是 Dubbo 的 Go 語言實(shí)現(xiàn)。 最近在 dubbo-go 的 todo list 上發(fā)現(xiàn),它還沒有實(shí)現(xiàn) TPS Limit 的模塊,于是就抽空
    發(fā)表于 03-17 15:27 ?738次閱讀

    詳解剖析Go語言調(diào)度模型的設(shè)計(jì)

    golang的MPG調(diào)度模型是保障Go語言效率高的一個(gè)重要特性,本文詳細(xì)介紹了Go語言調(diào)度模型的設(shè)計(jì)。 前言 Please remember that at the end of th
    的頭像 發(fā)表于 07-26 10:12 ?2355次閱讀
    詳解剖析<b class='flag-5'>Go</b>語言調(diào)度<b class='flag-5'>模型</b>的設(shè)計(jì)

    golang并發(fā)機(jī)制和其他語言在實(shí)現(xiàn)上有什么不同

    golang 并發(fā)機(jī)制和其他語言在實(shí)現(xiàn)上有什么不同?為什么能做到高效快速?本文做了詳細(xì)介紹。 由于對(duì)普通語法的介紹網(wǎng)上資源極多,Go 官方的上手指南 A Tour of Go: htt
    的頭像 發(fā)表于 07-29 16:35 ?1617次閱讀
    golang<b class='flag-5'>并發(fā)</b>機(jī)制和其他語言在<b class='flag-5'>實(shí)現(xiàn)</b>上有什么不同

    關(guān)于Actor并發(fā)模型的解析

    并發(fā)模型是用來實(shí)現(xiàn)不同應(yīng)用場(chǎng)景中并發(fā)任務(wù)的編程模型,通過合理地使用多線程,可以縮減應(yīng)用程序的開發(fā)和維護(hù)成本,同時(shí)還能更好地提升應(yīng)用程序在多核
    的頭像 發(fā)表于 07-18 09:23 ?2351次閱讀

    NVIDIA Triton 系列文章(10):模型并發(fā)執(zhí)行

    前面已經(jīng)做好了每個(gè)推理模型的基礎(chǔ)配置,基本上就能正常讓 Triton 服務(wù)器使用這些獨(dú)立模型進(jìn)行推理。接下來的重點(diǎn),就是要讓設(shè)備的計(jì)算資源盡可能地充分使用,首先第一件事情就是模型并發(fā)執(zhí)
    的頭像 發(fā)表于 01-05 11:55 ?1520次閱讀

    shell腳本實(shí)現(xiàn)并發(fā)多進(jìn)程

    在Shell腳本中實(shí)現(xiàn)并發(fā)多進(jìn)程可以使用以下方法: 使用符號(hào)來將其放入后臺(tái)執(zhí)行,從而實(shí)現(xiàn)并發(fā)多進(jìn)程。例如: #!/bin/bash command1 command3 符號(hào),它們將在后
    的頭像 發(fā)表于 11-08 10:20 ?1721次閱讀

    go語言如何解決并發(fā)問題

    作為一個(gè)后端開發(fā),日常工作中接觸最多的兩門語言就是PHP和GO了。無可否認(rèn),PHP確實(shí)是最好的語言(手動(dòng)狗頭哈哈),寫起來真的很舒爽,沒有任何心智負(fù)擔(dān),字符串和整型壓根就不用區(qū)分,開發(fā)速度真的是比
    的頭像 發(fā)表于 10-23 13:38 ?453次閱讀
    <b class='flag-5'>go</b>語言如何解決<b class='flag-5'>并發(fā)</b>問題