詳解Golang的GC三色標(biāo)記法
一 概念基礎(chǔ)
1.1三色標(biāo)記法將對(duì)象分為三類
把圖過(guò)程中遇到的對(duì)象,按“是否訪問(wèn)過(guò)”這個(gè)條件標(biāo)記成以下三種顏色:
白色對(duì)象(可能死亡):未被回收器訪問(wèn)到的對(duì)象。在回收開始階段,所有對(duì)象均為白色,當(dāng)回收結(jié)束后,白色對(duì)象均不可達(dá)。
灰色對(duì)象(波面):已被回收器訪問(wèn)到的對(duì)象,但回收器需要對(duì)其中的一個(gè)或多個(gè)指針進(jìn)行掃描,因?yàn)樗麄兛赡苓€指向白色對(duì)象。
黑色對(duì)象(確定存活):已被回收器訪問(wèn)到的對(duì)象,其中所有字段都已被掃描,黑色對(duì)象中任何一個(gè)指針都不可能直接指向白色對(duì)象。
1.2 標(biāo)記過(guò)程
- 起初所有的對(duì)象都是白色的;
- 從根對(duì)象出發(fā)掃描所有可達(dá)對(duì)象,標(biāo)記為灰色,放入待處理隊(duì)列;
- 從待處理隊(duì)列中取出灰色對(duì)象,將其引用的對(duì)象標(biāo)記為灰色并放入待處理隊(duì)列中,自身標(biāo)記為黑色;
- 重復(fù)步驟3,直到待處理隊(duì)列為空,此時(shí)白色對(duì)象即為不可達(dá)的“垃圾”,回收白色對(duì)象;
- 回收所有的白色對(duì)象,也就是回收垃圾
根對(duì)象
在垃圾回收的術(shù)語(yǔ)中又叫做根集合,它是垃圾回收器在標(biāo)記過(guò)程時(shí)最先檢查的對(duì)象。
- 全局變量:程序在編譯期就能確定的那些存在于程序整個(gè)生命周期的變量。
- 執(zhí)行棧:每個(gè) goroutine 都包含自己的執(zhí)行棧,這些執(zhí)行棧上包含棧上的變量及指向分配的堆內(nèi)存區(qū)塊的指針。
- 寄存器:寄存器的值可能表示一個(gè)指針,參與計(jì)算的這些指針可能指向某些賦值器分配的堆內(nèi)存區(qū)塊。
1.3 STW
STW(Stop The World)機(jī)制是指在進(jìn)行垃圾回收時(shí),會(huì)暫停應(yīng)用程序的運(yùn)行,以便進(jìn)行垃圾回收操作。這意味著在進(jìn)行垃圾回收時(shí),應(yīng)用程序?qū)o(wú)法繼續(xù)執(zhí)行。
為什么需要STW
STW(Stop-The-World)機(jī)制來(lái)確保并發(fā)操作的正確性。
如果不設(shè)置STW機(jī)制,那么在進(jìn)行GC時(shí),應(yīng)用程序線程可能會(huì)繼續(xù)執(zhí)行,從而導(dǎo)致內(nèi)存管理的不一致性和錯(cuò)誤。此外,GC可能會(huì)導(dǎo)致內(nèi)存分配和釋放的不連續(xù),從而導(dǎo)致內(nèi)存碎片化問(wèn)題。
因此,需要STW機(jī)制來(lái)確保GC的正確性和內(nèi)存管理的一致性。雖然STW機(jī)制會(huì)導(dǎo)致一定的性能損失,但是這是必要的代價(jià),以確保應(yīng)用程序的正確性和穩(wěn)定性。
1.4 屏障機(jī)制
1.4.1強(qiáng)、弱三色不變式
強(qiáng)三色不變式:強(qiáng)制性的不允許黑色對(duì)象引用白色對(duì)象,只能引用灰色對(duì)象,這樣就不會(huì)出現(xiàn)白色對(duì)象被誤刪的情況。
弱三色不等式 :
保護(hù)灰色對(duì)象到白色對(duì)象的路徑不會(huì)斷;
黑色對(duì)象可以引用白色對(duì)象,白色對(duì)象存在其他灰色對(duì)象對(duì)它的引用。
或可達(dá)它的鏈路上游存在灰色對(duì)象。這樣實(shí)則是黑色對(duì)象引用白色對(duì)象,白色對(duì)象處于一個(gè)被刪除的狀態(tài),但是上游灰色對(duì)象的引用,可以保護(hù)白色對(duì)象,使其安全。
為了遵循上述兩種方式,GC算法演進(jìn)到兩種屏障方式,“插入屏障”和“刪除屏障”。
1.4.2 插入屏障
在A對(duì)象引用B對(duì)象時(shí),B對(duì)象被標(biāo)記為灰色。(將B掛在A下游,B必須被標(biāo)記為灰色)
滿足強(qiáng)三色不等式。
插入屏障機(jī)制在??臻g的對(duì)象操作不使用,僅僅使用在堆空間對(duì)象的操作中。
1.4.3 刪除屏障
被刪除的對(duì)象,如果本身為灰色或白色,那么標(biāo)記為灰色。
滿足弱三色不等式。
1.4.4 混合屏障
插入寫屏障和刪除寫屏障的缺點(diǎn):
- 插入寫屏障:結(jié)束時(shí)需要STW來(lái)重新掃描棧,標(biāo)記棧上引用的白色對(duì)象存活
- 刪除寫屏障:回收精度低,GC開始時(shí)STW掃描堆棧來(lái)記錄快照,這個(gè)過(guò)程會(huì)保護(hù)開始時(shí)刻的所有的存活對(duì)象。
Go1.8引入混合寫屏障機(jī)制,避免了對(duì)棧的重復(fù)掃描過(guò)程,極大減少了STW的時(shí)間。
- GC開始將棧上的對(duì)象全部掃描并標(biāo)記為黑色(之后不再進(jìn)行第二次重復(fù)掃描,無(wú)需STW)
- GC期間,任何在棧上創(chuàng)建的新對(duì)象,都標(biāo)記為黑色
- 被刪除的對(duì)象標(biāo)記為灰色
- 被添加的對(duì)象標(biāo)記為灰色
二 GC過(guò)程
2.1 階段1:Mark Setup 標(biāo)記準(zhǔn)備
為了打開寫屏障,必須停止每個(gè)goroutine,讓垃圾收集器觀察并等待每個(gè)goroutine進(jìn)行函數(shù)調(diào)用,等待函數(shù)調(diào)用是為了保證goroutine停止時(shí)處于安全點(diǎn)。(期間會(huì)STW)
2.2 階段2:Marking 標(biāo)記
一旦寫屏障打開,垃圾收集器就開始標(biāo)記階段。
標(biāo)記階段需要標(biāo)記在堆內(nèi)存中仍然在使用中的值。首先檢查所有現(xiàn)goroutine的堆棧,以找到堆內(nèi)存的根指針。然后收集器必須從那些根指針遍歷堆內(nèi)存圖,標(biāo)記可以回收的內(nèi)存。
當(dāng)存在新的內(nèi)存分配時(shí),會(huì)暫停分配內(nèi)存過(guò)快的那些 goroutine,并將其轉(zhuǎn)去執(zhí)行一些輔助標(biāo)記(Mark Assist)的工作,從而達(dá)到放緩繼續(xù)分配、輔助 GC 的標(biāo)記工作的目的。
2.3階段3:Mark Termination 標(biāo)記結(jié)束
這個(gè)階段會(huì)關(guān)閉掉階段1開啟的屏障,并計(jì)算下一次清理的目標(biāo)和計(jì)劃。(本階段會(huì)STW)
2.4 階段4:Sweeping 清理
清理階段用于回收標(biāo)記階段中標(biāo)記出來(lái)的可回收內(nèi)存。當(dāng)應(yīng)用程序goroutine嘗試在堆內(nèi)存中分配新內(nèi)存時(shí),會(huì)觸發(fā)該操作,清理導(dǎo)致的延遲和吞吐量降低被分散到每次內(nèi)存分配時(shí)。
本階段會(huì)并發(fā)執(zhí)行,清除前面標(biāo)記出來(lái)需清理的內(nèi)存。
以上就是詳解Golang的GC三色標(biāo)記法的詳細(xì)內(nèi)容,更多關(guān)于Golang GC 三色標(biāo)記法的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
goland?-sync/atomic原子操作小結(jié)
這篇文章主要介紹了goland?-sync/atomic原子操作,原子操作能夠保證執(zhí)行期間是連續(xù)且不會(huì)被中斷(變量不會(huì)被其他修改,mutex可能存在被其他修改的情況),本文給大家介紹的非常詳細(xì),需要的朋友參考下2022-08-08
Go可變參數(shù)函數(shù)的實(shí)現(xiàn)
可變參數(shù)函數(shù)是指函數(shù)參數(shù)的某個(gè)參數(shù)可有可無(wú),即這個(gè)參數(shù)的個(gè)數(shù)可以為0會(huì)多個(gè),可變參數(shù)函數(shù)參數(shù)在日常編程中大量使用,本文主要介紹了Go可變參數(shù)函數(shù)的實(shí)現(xiàn),感興趣的可以了解一下2023-12-12
Go語(yǔ)言基礎(chǔ)go install命令使用示例詳解
這篇文章主要為大家介紹了Go語(yǔ)言基礎(chǔ)go install命令的使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2021-11-11
GoLang之使用Context控制請(qǐng)求超時(shí)的實(shí)現(xiàn)
這篇文章主要介紹了GoLang之使用Context控制請(qǐng)求超時(shí)的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04
Golang 關(guān)于Gin框架請(qǐng)求參數(shù)的獲取方法
Gin是Go語(yǔ)言的Web框架,提供路由和中間件支持,本文介紹如何使用Gin獲取HTTP請(qǐng)求參數(shù),包括URLPath參數(shù)、URLQuery參數(shù)、HTTPBody參數(shù)和Header參數(shù),詳解直接獲取和綁定到結(jié)構(gòu)體兩種方法,幫助開發(fā)者高效處理Web請(qǐng)求2024-10-10
Go語(yǔ)言LeetCode題解706設(shè)計(jì)哈希映射
這篇文章主要為大家介紹了Go語(yǔ)言LeetCode題解706設(shè)計(jì)哈希映射示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12
基于Go+WebSocket實(shí)現(xiàn)實(shí)時(shí)通信功能
在互聯(lián)網(wǎng)應(yīng)用程序中,實(shí)時(shí)通信是一種非常重要的功能,WebSocket 是一種基于 TCP 的協(xié)議,它允許客戶端和服務(wù)器之間進(jìn)行雙向通信,本文將介紹如何使用 Golang 創(chuàng)建單獨(dú)的 WebSocket 會(huì)話,以實(shí)現(xiàn)實(shí)時(shí)通信功能,需要的朋友可以參考下2023-10-10

