Go 的select語(yǔ)句是一種僅能用于channl發(fā)送和接收消息的專(zhuān)用語(yǔ)句,此語(yǔ)句運(yùn)行期間是阻塞的;當(dāng)select中沒(méi)有case語(yǔ)句的時(shí)候,會(huì)阻塞當(dāng)前的groutine。所以,有人也會(huì)說(shuō)select是用來(lái)阻塞監(jiān)聽(tīng)goroutine的。
成都創(chuàng)新互聯(lián)是一家專(zhuān)注于成都網(wǎng)站制作、成都網(wǎng)站建設(shè)與策劃設(shè)計(jì),思禮網(wǎng)站建設(shè)哪家好?成都創(chuàng)新互聯(lián)做網(wǎng)站,專(zhuān)注于網(wǎng)站建設(shè)10多年,網(wǎng)設(shè)計(jì)領(lǐng)域的專(zhuān)業(yè)建站公司;建站業(yè)務(wù)涵蓋:思禮等地區(qū)。思禮做網(wǎng)站價(jià)格咨詢(xún):18980820575
還有人說(shuō):select是Golang在語(yǔ)言層面提供的I/O多路復(fù)用的機(jī)制,其專(zhuān)門(mén)用來(lái)檢測(cè)多個(gè)channel是否準(zhǔn)備完畢:可讀或可寫(xiě)。
以上說(shuō)法都正確。
我們來(lái)回顧一下是什么是 I/O多路復(fù)用 。
每來(lái)一個(gè)進(jìn)程,都會(huì)建立連接,然后阻塞,直到接收到數(shù)據(jù)返回響應(yīng)。
普通這種方式的缺點(diǎn)其實(shí)很明顯:系統(tǒng)需要?jiǎng)?chuàng)建和維護(hù)額外的線(xiàn)程或進(jìn)程。因?yàn)榇蠖鄶?shù)時(shí)候,大部分阻塞的線(xiàn)程或進(jìn)程是處于等待狀態(tài),只有少部分會(huì)接收并處理響應(yīng),而其余的都在等待。系統(tǒng)為此還需要多做很多額外的線(xiàn)程或者進(jìn)程的管理工作。
為了解決圖中這些多余的線(xiàn)程或者進(jìn)程,于是有了"I/O多路復(fù)用"
每個(gè)線(xiàn)程或者進(jìn)程都先到圖中”裝置“中注冊(cè),然后阻塞,然后只有一個(gè)線(xiàn)程在”運(yùn)輸“,當(dāng)注冊(cè)的線(xiàn)程或者進(jìn)程準(zhǔn)備好數(shù)據(jù)后,”裝置“會(huì)根據(jù)注冊(cè)的信息得到相應(yīng)的數(shù)據(jù)。從始至終kernel只會(huì)使用圖中這個(gè)黃黃的線(xiàn)程,無(wú)需再對(duì)額外的線(xiàn)程或者進(jìn)程進(jìn)行管理,提升了效率。
select的實(shí)現(xiàn)經(jīng)歷了多個(gè)版本的修改,當(dāng)前版本為:1.11
select這個(gè)語(yǔ)句底層實(shí)現(xiàn)實(shí)際上主要由兩部分組成: case語(yǔ)句 和 執(zhí)行函數(shù) 。
源碼地址為:/go/src/runtime/select.go
每個(gè)case語(yǔ)句,單獨(dú)抽象出以下結(jié)構(gòu)體:
結(jié)構(gòu)體可以用下圖表示:
然后執(zhí)行select語(yǔ)句實(shí)際上就是調(diào)用 func selectgo(cas0 *scase, order0 *uint16, ncases int) (int, bool) 函數(shù)。
func selectgo(cas0 *scase, order0 *uint16, ncases int) (int, bool) 函數(shù)參數(shù):
selectgo 返回所選scase的索引(該索引與其各自的select {recv,send,default}調(diào)用的序號(hào)位置相匹配)。此外,如果選擇的scase是接收操作(recv),則返回是否接收到值。
誰(shuí)負(fù)責(zé)調(diào)用 func selectgo(cas0 *scase, order0 *uint16, ncases int) (int, bool) 函數(shù)呢?
在 /reflect/value.go 中有個(gè) func rselect([]runtimeSelect) (chosen int, recvOK bool) 函數(shù),此函數(shù)的實(shí)現(xiàn)在 /runtime/select.go 文件中的 func reflect_rselect(cases []runtimeSelect) (int, bool) 函數(shù)中:
那誰(shuí)調(diào)用的 func rselect([]runtimeSelect) (chosen int, recvOK bool) 呢?
在 /refect/value.go 中,有一個(gè) func Select(cases []SelectCase) (chosen int, recv Value, recvOK bool) 的函數(shù),其調(diào)用了 rselect 函數(shù),并將最終Go中select語(yǔ)句的返回值的返回。
以上這三個(gè)函數(shù)的調(diào)用棧按順序如下:
這仨函數(shù)中無(wú)論是返回值還是參數(shù)都大同小異,可以簡(jiǎn)單粗暴的認(rèn)為:函數(shù)參數(shù)傳入的是case語(yǔ)句,返回值返回被選中的case語(yǔ)句。
那誰(shuí)調(diào)用了 func Select(cases []SelectCase) (chosen int, recv Value, recvOK bool) 呢?
可以簡(jiǎn)單的認(rèn)為是系統(tǒng)了。
來(lái)個(gè)簡(jiǎn)單的圖:
前兩個(gè)函數(shù) Select 和 rselect 都是做了簡(jiǎn)單的初始化參數(shù),調(diào)用下一個(gè)函數(shù)的操作。select真正的核心功能,是在最后一個(gè)函數(shù) func selectgo(cas0 *scase, order0 *uint16, ncases int) (int, bool) 中實(shí)現(xiàn)的。
打亂傳入的case結(jié)構(gòu)體順序
鎖住其中的所有的channel
遍歷所有的channel,查看其是否可讀或者可寫(xiě)
如果其中的channel可讀或者可寫(xiě),則解鎖所有channel,并返回對(duì)應(yīng)的channel數(shù)據(jù)
假如沒(méi)有channel可讀或者可寫(xiě),但是有default語(yǔ)句,則同上:返回default語(yǔ)句對(duì)應(yīng)的scase并解鎖所有的channel。
假如既沒(méi)有channel可讀或者可寫(xiě),也沒(méi)有default語(yǔ)句,則將當(dāng)前運(yùn)行的groutine阻塞,并加入到當(dāng)前所有channel的等待隊(duì)列中去。
然后解鎖所有channel,等待被喚醒。
此時(shí)如果有個(gè)channel可讀或者可寫(xiě)ready了,則喚醒,并再次加鎖所有channel,
遍歷所有channel找到那個(gè)對(duì)應(yīng)的channel和G,喚醒G,并將沒(méi)有成功的G從所有channel的等待隊(duì)列中移除。
如果對(duì)應(yīng)的scase值不為空,則返回需要的值,并解鎖所有channel
如果對(duì)應(yīng)的scase為空,則循環(huán)此過(guò)程。
在想想select和channel做了什么事兒,我覺(jué)得和多路復(fù)用是一回事兒
Go語(yǔ)言是谷歌推出的一種全新的編程語(yǔ)言,可以在不損失應(yīng)用程序性能的情況下降低代碼的復(fù)雜性。谷歌首席軟件工程師羅布派克(Rob Pike)說(shuō):我們之所以開(kāi)發(fā)Go,是因?yàn)檫^(guò)去10多年間軟件開(kāi)發(fā)的難度令人沮喪。
Go是谷歌2009發(fā)布的第二款編程語(yǔ)言。2009年7月份,谷歌曾發(fā)布了Simple語(yǔ)言,它是用來(lái)開(kāi)發(fā)Android應(yīng)用的一種BASIC語(yǔ)言.
北京時(shí)間2010年1月10日,Go語(yǔ)言摘得了TIOBE公布的2009年年度大獎(jiǎng)。該獎(jiǎng)項(xiàng)授予在2009年市場(chǎng)份額增長(zhǎng)最多的編程語(yǔ)言。
谷歌資深軟件工程師羅布·派克(Rob Pike)表示,"Go讓我體驗(yàn)到了從未有過(guò)的開(kāi)發(fā)效率。"派克表示,今天的C++或C一樣,Go是一種系統(tǒng)語(yǔ)言。他解釋道,"使用它可以進(jìn)行快速開(kāi)發(fā),同時(shí)它還是一個(gè)真正的編譯語(yǔ)言,我們之所以現(xiàn)在將其開(kāi)源,原因是我們認(rèn)為它已經(jīng)非常有用和強(qiáng)大。"
2007年,谷歌把Go作為一個(gè)20%項(xiàng)目開(kāi)始研發(fā),即讓員工抽出本職工作之外時(shí)間的20%, 投入在該項(xiàng)目上。除了派克外,該項(xiàng)目的成員還有其他谷歌工程師也參與研發(fā)。
派克表示,編譯后Go代碼的運(yùn)行速度與C語(yǔ)言非常接近,而且編譯速度非常快,就像在使用一個(gè)交互式語(yǔ)言。現(xiàn)有編程語(yǔ)言均未專(zhuān)門(mén)對(duì)多核處理器進(jìn)行優(yōu)化。Go就是谷歌工程師為這類(lèi)程序編寫(xiě)的一種語(yǔ)言。它不是針對(duì)編程初學(xué)者設(shè)計(jì)的,但學(xué)習(xí)使用它也不是非常困難。Go支持面向?qū)ο?,而且具有真正的閉包(closures)和反射 (reflection)等功能。
在學(xué)習(xí)曲線(xiàn)方面,派克認(rèn)為Go與Java類(lèi)似,對(duì)于Java開(kāi)發(fā)者來(lái)說(shuō),應(yīng)該能夠輕松學(xué)會(huì) Go。之所以將Go作為一個(gè)開(kāi)源項(xiàng)目發(fā)布,目的是讓開(kāi)源社區(qū)有機(jī)會(huì)創(chuàng)建更好的工具來(lái)使用該語(yǔ)言,例如 Eclipse IDE中的插件。
在谷歌公開(kāi)發(fā)布的所有網(wǎng)絡(luò)應(yīng)用中,均沒(méi)有使用Go,但是谷歌已經(jīng)使用該語(yǔ)言開(kāi)發(fā)了幾個(gè)內(nèi)部項(xiàng)目。派克表示,Go是否會(huì)對(duì)谷歌即將推出的Chrome OS產(chǎn)生影響,還言之尚早,不過(guò)Go的確可以和Native Client配合使用。他表示"Go可以讓?xiě)?yīng)用完美的運(yùn)行在瀏覽器內(nèi)。"例如,使用Go可以更高效的實(shí)現(xiàn)Wave,無(wú)論是在前端還是后臺(tái)。
Go 同時(shí)具有兩種編譯器,一種是建立在GCC基礎(chǔ)上的Gccgo,另外一種是分別針對(duì)64位x64和32位x86計(jì)算機(jī)的一套編譯器(6g和8g)。谷歌目前正在研發(fā)其對(duì)ARM芯片和Android設(shè)備的支持。派克表示,"Android手機(jī)存在的問(wèn)題是,我們一直沒(méi)有一個(gè)數(shù)學(xué)協(xié)處理器。"
GO語(yǔ)言由Google公司開(kāi)發(fā),并于2009年開(kāi)源,對(duì)比Java、Python、C等語(yǔ)言,GO尤其擅長(zhǎng)并發(fā)編程,性能堪比C語(yǔ)言,開(kāi)發(fā)效率比肩Python,被譽(yù)為21世紀(jì)的C語(yǔ)言。GO語(yǔ)言在云計(jì)算、大數(shù)據(jù)、微服務(wù)、高并發(fā)領(lǐng)域,應(yīng)用非常廣泛。BAT大廠(chǎng)正在把GO作為新項(xiàng)目開(kāi)發(fā)的首選語(yǔ)言。
Go runtime的調(diào)度器:
在了解Go的運(yùn)行時(shí)的scheduler之前,需要先了解為什么需要它,因?yàn)槲覀兛赡軙?huì)想,OS內(nèi)核不是已經(jīng)有一個(gè)線(xiàn)程scheduler了嘛?
熟悉POSIX API的人都知道,POSIX的方案在很大程度上是對(duì)Unix process進(jìn)場(chǎng)模型的一個(gè)邏輯描述和擴(kuò)展,兩者有很多相似的地方。 Thread有自己的信號(hào)掩碼,CPU affinity等。但是很多特征對(duì)于Go程序來(lái)說(shuō)都是累贅。 尤其是context上下文切換的耗時(shí)。另一個(gè)原因是Go的垃圾回
Go語(yǔ)言也稱(chēng) Golang,兼具效率、性能、安全、健壯等特性。這套Go語(yǔ)言教程(Golang教程)通俗易懂,深入淺出,既適合沒(méi)有基礎(chǔ)的讀者快速入門(mén),也適合工作多年的程序員查閱知識(shí)點(diǎn)。
Go 語(yǔ)言
這套教程在講解一些知識(shí)點(diǎn)時(shí),將 Go 語(yǔ)言和其他多種語(yǔ)言進(jìn)行對(duì)比,讓掌握其它編程語(yǔ)言的讀者能迅速理解 Go 語(yǔ)言的特性。Go語(yǔ)言從底層原生支持并發(fā),無(wú)須第三方庫(kù)、開(kāi)發(fā)者的編程技巧和開(kāi)發(fā)經(jīng)驗(yàn)就可以輕松搞定。
Go語(yǔ)言(或 Golang)起源于 2007 年,并在 2009 年正式對(duì)外發(fā)布。Go 是非常年輕的一門(mén)語(yǔ)言,它的主要目標(biāo)是“兼具 Python 等動(dòng)態(tài)語(yǔ)言的開(kāi)發(fā)速度和 C/C++ 等編譯型語(yǔ)言的性能與安全性”。
Go語(yǔ)言是編程語(yǔ)言設(shè)計(jì)的又一次嘗試,是對(duì)類(lèi)C語(yǔ)言的重大改進(jìn),它不但能讓你訪(fǎng)問(wèn)底層操作系統(tǒng),還提供了強(qiáng)大的網(wǎng)絡(luò)編程和并發(fā)編程支持。Go語(yǔ)言的用途眾多,可以進(jìn)行網(wǎng)絡(luò)編程、系統(tǒng)編程、并發(fā)編程、分布式編程。
Go語(yǔ)言的推出,旨在不損失應(yīng)用程序性能的情況下降低代碼的復(fù)雜性,具有“部署簡(jiǎn)單、并發(fā)性好、語(yǔ)言設(shè)計(jì)良好、執(zhí)行性能好”等優(yōu)勢(shì),目前國(guó)內(nèi)諸多 IT 公司均已采用Go語(yǔ)言開(kāi)發(fā)項(xiàng)目。Go語(yǔ)言有時(shí)候被描述為“C 類(lèi)似語(yǔ)言”,或者是“21 世紀(jì)的C語(yǔ)言”。Go 從C語(yǔ)言繼承了相似的表達(dá)式語(yǔ)法、控制流結(jié)構(gòu)、基礎(chǔ)數(shù)據(jù)類(lèi)型、調(diào)用參數(shù)傳值、指針等很多思想,還有C語(yǔ)言一直所看中的編譯后機(jī)器碼的運(yùn)行效率以及和現(xiàn)有操作系統(tǒng)的無(wú)縫適配。
因?yàn)镚o語(yǔ)言沒(méi)有類(lèi)和繼承的概念,所以它和 Java 或 C++ 看起來(lái)并不相同。但是它通過(guò)接口(interface)的概念來(lái)實(shí)現(xiàn)多態(tài)性。Go語(yǔ)言有一個(gè)清晰易懂的輕量級(jí)類(lèi)型系統(tǒng),在類(lèi)型之間也沒(méi)有層級(jí)之說(shuō)。因此可以說(shuō)Go語(yǔ)言是一門(mén)混合型的語(yǔ)言。
此外,很多重要的開(kāi)源項(xiàng)目都是使用Go語(yǔ)言開(kāi)發(fā)的,其中包括 Docker、Go-Ethereum、Thrraform 和 Kubernetes。Go 是編譯型語(yǔ)言,Go 使用編譯器來(lái)編譯代碼。編譯器將源代碼編譯成二進(jìn)制(或字節(jié)碼)格式;在編譯代碼時(shí),編譯器檢查錯(cuò)誤、優(yōu)化性能并輸出可在不同平臺(tái)上運(yùn)行的二進(jìn)制文件。要?jiǎng)?chuàng)建并運(yùn)行 Go 程序,程序員必須執(zhí)行如下步驟。
使用文本編輯器創(chuàng)建 Go 程序;
保存文件;編譯程序;運(yùn)行編譯得到的可執(zhí)行文件。
這不同于 Python、Ruby 和 JavaScript 等語(yǔ)言,它們不包含編譯步驟。Go 自帶了編譯器,因此無(wú)須單獨(dú)安裝編譯器。
鏈喬教育在線(xiàn)旗下學(xué)碩創(chuàng)新區(qū)塊鏈技術(shù)工作站是中國(guó)教育部學(xué)校規(guī)劃建設(shè)發(fā)展中心開(kāi)展的“智慧學(xué)習(xí)工場(chǎng)2020-學(xué)碩創(chuàng)新工作站 ”唯一獲準(zhǔn)的“區(qū)塊鏈技術(shù)專(zhuān)業(yè)”試點(diǎn)工作站。專(zhuān)業(yè)站立足為學(xué)生提供多樣化成長(zhǎng)路徑,推進(jìn)專(zhuān)業(yè)學(xué)位研究生產(chǎn)學(xué)研結(jié)合培養(yǎng)模式改革,構(gòu)建應(yīng)用型、復(fù)合型人才培養(yǎng)體系。