gc 與gccgo 都是go語言標(biāo)準(zhǔn)規(guī)范的不同實(shí)現(xiàn),兩者包含不同的側(cè)重點(diǎn):
創(chuàng)新互聯(lián)堅(jiān)持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:網(wǎng)站設(shè)計(jì)制作、成都網(wǎng)站制作、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿足客戶于互聯(lián)網(wǎng)時(shí)代的包頭網(wǎng)站設(shè)計(jì)、移動(dòng)媒體設(shè)計(jì)的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!
使用成本上gccgo遠(yuǎn)比gc更高,基于如下原因:
總結(jié):除非真要追求高性能,否則不建議去折騰gccgo
如果一定要折騰,建議思路:基于gcc docker 鏡像,編寫Dockerfile,安裝golang,然后使用 go build -compiler=gccgo 。
相關(guān)資源:
Go語言(又稱 Golang)是 Google 的 Robert Griesemer,Rob Pike 及 Ken Thompson 開發(fā)的一種靜態(tài)強(qiáng)類型、編譯型語言。Go 語言語法與 C 相近,但功能上有:內(nèi)存安全,GC(垃圾回收),結(jié)構(gòu)形態(tài)及 CSP-style 并發(fā)計(jì)算。 擴(kuò)展資料
Go語言主要用作服務(wù)器端開發(fā),其定位是用來開發(fā)“大型軟件”的,適合于很多程序員一起開發(fā)大型軟件,并且開發(fā)周期長,支持云計(jì)算的網(wǎng)絡(luò)服務(wù)。Go語言能夠讓程序員快速開發(fā),并且在軟件不斷的'增長過程中,它能讓程序員更容易地進(jìn)行維護(hù)和修改。它融合了傳統(tǒng)編譯型語言的高效性和腳本語言的易用性和富于表達(dá)性。
Go語言作為服務(wù)器編程語言,很適合處理日志、數(shù)據(jù)打包、虛擬機(jī)處理、文件系統(tǒng)、分布式系統(tǒng)、數(shù)據(jù)庫代理等;網(wǎng)絡(luò)編程方面,Go語言廣泛應(yīng)用于Web應(yīng)用、API應(yīng)用、下載應(yīng)用等;除此之外,Go語言還可用于內(nèi)存數(shù)據(jù)庫和云平臺領(lǐng)域,目前國外很多云平臺都是采用Go開發(fā)。
Golang采用了三色標(biāo)記法來進(jìn)行垃圾回收,那么在什么場景下會(huì)觸發(fā)這個(gè)回收動(dòng)作呢?
源碼主要位于文件 src/runtime/mgc.go go version 1.16
觸發(fā)條件從大方面說,可分為 手動(dòng)觸發(fā) 和 系統(tǒng)觸發(fā) 兩種方式。手動(dòng)觸發(fā)一般很少用,主要由開發(fā)者通過調(diào)用 runtime.GC() 函數(shù)來實(shí)現(xiàn),而對于系統(tǒng)自動(dòng)觸發(fā)是 運(yùn)行時(shí) 根據(jù)一些條件判斷來進(jìn)行的,這也正是本文要介紹的內(nèi)容。
不管哪種觸發(fā)方式,底層回收機(jī)制是一樣的,所以我們先看一下手動(dòng)觸發(fā),根據(jù)它來找系統(tǒng)觸發(fā)的條件。
可以看到開始執(zhí)行GC的是 gcStart() 函數(shù),它有一個(gè) gcTrigger 參數(shù),是一個(gè)觸發(fā)條件結(jié)構(gòu)體,它的結(jié)構(gòu)體也很簡單。
其實(shí)在Golang 內(nèi)部所有的GC都是通過 gcStart() 函數(shù),然后指定一個(gè) gcTrigger 的參數(shù)來開始的,而手動(dòng)觸發(fā)指定的條件值為 gcTriggerCycle 。 gcStart 是一個(gè)很復(fù)雜的函數(shù),有興趣的可以看一下源碼實(shí)現(xiàn)。
對于 kind 的值有三種,分別為 gcTriggerHeap 、 gcTriggerTime 和 gcTriggerCycle 。
運(yùn)行時(shí)會(huì)通過 gcTrigger.test() 函數(shù)來決定是否需要觸發(fā)GC,只要滿足上面基中一個(gè)即可。
到此我們基本明白了這三種觸發(fā)GC的條件,那么對于系統(tǒng)自動(dòng)觸發(fā)這種,Golang 從一個(gè)程序的開始到運(yùn)行,它又是如何一步一步監(jiān)控到這個(gè)條件的呢?
其實(shí) runtime 在程序啟動(dòng)時(shí),會(huì)在一個(gè)初始化函數(shù) init() 里啟用一個(gè) forcegchelper() 函數(shù),這個(gè)函數(shù)位于 proc.go 文件。
為了減少系統(tǒng)資源占用,在 forcegchelper 函數(shù)里會(huì)通過 goparkunlock() 函數(shù)主動(dòng)讓自己陷入休眠,以后由 sysmon() 監(jiān)控線程根據(jù)條件來恢復(fù)這個(gè)gc goroutine。
可以看到 sysmon() 會(huì)在一個(gè) for 語句里一直判斷這個(gè) gcTriggerTime 這個(gè)條件是否滿足,如果滿足的話,會(huì)將 forcegc.g 這個(gè) goroutine 添加到全局隊(duì)列里進(jìn)行調(diào)度(這里 forcegc 是一個(gè)全局變量)。
調(diào)度器在調(diào)度循環(huán) runtime.schedule 中還可以通過垃圾收集控制器的 runtime.gcControllerState.findRunnabledGCWorker 獲取并執(zhí)行用于后臺標(biāo)記的任務(wù)。
正如sycn.Pool的名字所示,這是go中實(shí)現(xiàn)的一個(gè)對象池,為什么要有這個(gè)池呢?首先go是自帶垃圾回收機(jī)制(也就是通常所說的gc)。gc會(huì)帶來運(yùn)行時(shí)的開銷,對于高頻的內(nèi)存申請與釋放,如果將不用的對象存放在一個(gè)池子中,用的時(shí)候從池子中取出一個(gè)對象,用完了再還回去,這樣就能減輕gc的壓力。
對于池這個(gè)概念,之前可能聽說過連接池。能否用sync.Pool實(shí)現(xiàn)一個(gè)連接池呢?答案是不能的。因?yàn)閷τ趕ync.Pool而言,我們無法保證每次放回去再取出來的對象是與之前一致的,對象的內(nèi)存存在著唄銷毀的可能。因此,這個(gè)sync.Pool的存在僅僅是為了減緩gc的壓力而生的。
定義sync.Pool的時(shí)候只需要設(shè)置一個(gè)New成員,它是一個(gè)函數(shù),類型為func() interface{},當(dāng)池子中沒有空閑的對象時(shí)就會(huì)調(diào)用New函數(shù)生成一個(gè)。由于pool中對象的數(shù)量不可控,因此并沒有傳遞任何與對象數(shù)量有關(guān)的參數(shù)。
然后,調(diào)用調(diào)用Get函數(shù)就可以取出一個(gè)對象,調(diào)用Put函數(shù)就可以將對象歸還到池子中。