欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Golang內(nèi)存泄露場景與定位方式的實現(xiàn)

 更新時間:2024年04月19日 09:41:56   作者:每天一個禿頂小技巧  
Golang有自動垃圾回收機制,但是仍然可能會出現(xiàn)內(nèi)存泄漏的情況,本文主要介紹了Golang內(nèi)存泄露場景與定位方式的實現(xiàn),具有一定的參考價值,感興趣的可以了解一下

一、產(chǎn)生原因

Golang有自動垃圾回收機制,但是仍然可能會出現(xiàn)內(nèi)存泄漏的情況。以下是Golang內(nèi)存泄漏的常見可能原因:

  • 循環(huán)引用:如果兩個或多個對象相互引用,且沒有其他對象引用它們,那么它們就會被垃圾回收機制誤認為是仍在使用的對象,導致內(nèi)存泄漏。
  • 全局變量:在Golang中,全局變量的生命周期與程序的生命周期相同。如果一個全局變量被創(chuàng)建后一直存在于內(nèi)存中,那么它所占用的內(nèi)存就無法被回收,可能會導致內(nèi)存泄漏。
  • 未關(guān)閉的文件句柄:如果程序打開了文件句柄但沒有關(guān)閉它們,那么這些文件句柄所占用的內(nèi)存就無法被回收,可能會導致內(nèi)存泄漏。
  • 大量的臨時對象:如果程序創(chuàng)建了大量的臨時對象,但沒有及時釋放它們,那么這些對象所占用的內(nèi)存就無法被回收,可能會導致內(nèi)存泄漏。
  • goroutine泄漏常見的泄露場景,例如協(xié)程發(fā)生阻塞,Go運行時并不會將處于永久阻塞狀態(tài)的協(xié)程殺掉,因此永久處于阻塞狀態(tài)的協(xié)程所占用的資源將永得不到釋放。
  • time.Ticker未關(guān)閉導致泄漏:當一個time.Timer值不再被使用,一段時間后它將被自動垃圾回收掉。 但對于一個不再使用的time.Ticker值,我們必須調(diào)用它的Stop方法結(jié)束它,否則它將永遠不會得到回收。

二、排查方式

如果出現(xiàn)內(nèi)存泄漏,可以使用以下方式進行分析,找出內(nèi)存泄漏的原因并進行修復。

  • 使用 Go 語言自帶的 pprof 工具進行分析。pprof 可以生成程序的 CPU 和內(nèi)存使用情況的報告,幫助開發(fā)者找出程序中的性能瓶頸和內(nèi)存泄漏問題??梢酝ㄟ^在代碼中添加 import _ "net/http/pprof" 和 http.ListenAndServe("localhost:6060", nil) 來開啟 pprof 工具。
  • 使用 Golang 內(nèi)置的 runtime 包進行分析。runtime 包提供了一些函數(shù),包括 SetFinalizerReadMemStats 和 Stack 等,可以幫助開發(fā)者了解程序的內(nèi)存使用情況和內(nèi)存泄漏問題。
  • 使用第三方工具進行分析。例如,可以使用 go-torch 工具生成火焰圖,幫助開發(fā)者找出程序中的性能瓶頸和內(nèi)存泄漏問題。
  • 使用 go vet 工具進行靜態(tài)分析。go vet 可以檢查程序中的常見錯誤和潛在問題,包括內(nèi)存泄漏問題。
  • 代碼審查。開發(fā)者可以通過代碼審查來找出程序中的潛在問題和內(nèi)存泄漏問題。

三、通過 pprof 的命令排查內(nèi)存泄露問題

3.1 通過 pprof 的命令行分析 heap

命令行執(zhí)行命令: go tool pprof -inuse_space [<http://127.0.0.1:9999/debug/pprof/heap>](<http://spark-master.x.upyun.com/debug/pprof/heap>)

這個命令的作用是, 抓取當前程序已使用的 heap. 抓取后, 就可以進行類似于 gdb 的交互操作.

top 命令, 默認能列出當前程序中內(nèi)存占用排名前 10 的函數(shù). 如圖. 當時進行到這一步的時候, 我就非常驚訝, 因為 time.NewTimer 居然占據(jù)了 6 個多 G 的內(nèi)存.

img

list <函數(shù)名>, 展現(xiàn)函數(shù)內(nèi)部的內(nèi)存占用. 使用 list time.NewTimer 查看了該函數(shù)的內(nèi)部, 真相大白了, 原來每次調(diào)用 NewTimer 都會創(chuàng)建一個 channel, 還會生成一個結(jié)構(gòu)體 runtimeTimer, 應該就是這兩個地方內(nèi)存沒有釋放造成的內(nèi)存泄露.

img

3.2 修改 for ... select ... time.After 造成的內(nèi)存泄露

原來程序中存在如下代碼:

for {
		select {

		case a := <-chanA:
			...

		case b := <-chanB:
			....

		case <-time.After(20*time.Minutes):
			return nil, errors.New("download timeout")
	}

time.After 就是封裝了一層的 NewTimertime.After 的源碼:

func After(d Duration) <-chan Time {
	return NewTimer(d).C
}

修復該錯誤, 只調(diào)用一次 NewTimer:

downloadTimeout := time.NewTimer(20 * time.Minute)
// 添加關(guān)閉時退出操作
defer downloadTimeout.Stop()

for {
		select {

		case a := <-chanA:
			...

		case b := <-chanB:
			....

		case <-downloadTimeout.C:
			return nil, errors.New("download timeout")
	}

四、總結(jié)

通過這篇文章我們了解到Golang內(nèi)存泄漏的常見可能原因有哪些:

  • 循環(huán)引用:如果兩個或多個對象相互引用,且沒有其他對象引用它們,那么它們就會被垃圾回收機制誤認為是仍在使用的對象,導致內(nèi)存泄漏。
  • 全局變量:在Golang中,全局變量的生命周期與程序的生命周期相同。如果一個全局變量被創(chuàng)建后一直存在于內(nèi)存中,那么它所占用的內(nèi)存就無法被回收,可能會導致內(nèi)存泄漏。
  • 未關(guān)閉的文件句柄:如果程序打開了文件句柄但沒有關(guān)閉它們,那么這些文件句柄所占用的內(nèi)存就無法被回收,可能會導致內(nèi)存泄漏。
  • 大量的臨時對象:如果程序創(chuàng)建了大量的臨時對象,但沒有及時釋放它們,那么這些對象所占用的內(nèi)存就無法被回收,可能會導致內(nèi)存泄漏。
  • goroutine泄漏:比較常見的泄露場景,例如協(xié)程發(fā)生阻塞,Go運行時并不會將處于永久阻塞狀態(tài)的協(xié)程殺掉,因此永久處于阻塞狀態(tài)的協(xié)程所占用的資源將永得不到釋放。
  • time.Ticker未關(guān)閉導致泄漏:當一個time.Timer值不再被使用,一段時間后它將被自動垃圾回收掉。 但對于一個不再使用的time.Ticker值,我們必須調(diào)用它的Stop方法結(jié)束它,否則它將永遠不會得到回收。

然后介紹了相關(guān)排查工具以及pprof如何排查內(nèi)存泄露問題。

五、參考鏈接

1.一些可能的內(nèi)存泄漏場景

2.使用 pprof 排查 Golang 內(nèi)存泄露

到此這篇關(guān)于Golang內(nèi)存泄露場景與定位方式的實現(xiàn)的文章就介紹到這了,更多相關(guān)Golang內(nèi)存泄露內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 云端golang開發(fā),無需本地配置,能上網(wǎng)就能開發(fā)和運行

    云端golang開發(fā),無需本地配置,能上網(wǎng)就能開發(fā)和運行

    這篇文章主要介紹了云端golang開發(fā),無需本地配置,能上網(wǎng)就能開發(fā)和運行的相關(guān)資料,需要的朋友可以參考下
    2023-10-10
  • Golang使用泛型對數(shù)組進行去重的實現(xiàn)

    Golang使用泛型對數(shù)組進行去重的實現(xiàn)

    本文主要介紹了Golang使用泛型對數(shù)組進行去重的實現(xiàn),通過使用類型參數(shù)T和類型約束any,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2025-02-02
  • Go中的go.mod使用詳解

    Go中的go.mod使用詳解

    這篇文章主要介紹了Go中的go.mod使用方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-12-12
  • Go語言操作Excel利器之excelize類庫詳解

    Go語言操作Excel利器之excelize類庫詳解

    Excelize是Go語言編寫的用于操作Office Excel文檔基礎(chǔ)庫,基于ECMA-376,ISO/IEC 29500國際標準,可以使用它來讀取、寫入由Excel 2007及以上版本創(chuàng)建的電子表格文檔,下面這篇文章主要給大家介紹了關(guān)于Go語言操作Excel利器之excelize類庫的相關(guān)資料,需要的朋友可以參考下
    2022-10-10
  • golang中單機鎖的具體實現(xiàn)詳解

    golang中單機鎖的具體實現(xiàn)詳解

    這篇文章主要為大家詳細介紹了golang中單機鎖的具體實現(xiàn)的相關(guān)知識,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下
    2025-03-03
  • Go語言數(shù)據(jù)結(jié)構(gòu)之單鏈表的實例詳解

    Go語言數(shù)據(jù)結(jié)構(gòu)之單鏈表的實例詳解

    鏈表由一系列結(jié)點(鏈表中每一個元素稱為結(jié)點)組成,結(jié)點可以在運行時動態(tài)生成。本文將通過五個例題帶大家深入了解Go語言中單鏈表的用法,感興趣的可以了解一下
    2022-08-08
  • golang?chan傳遞數(shù)據(jù)的性能開銷詳解

    golang?chan傳遞數(shù)據(jù)的性能開銷詳解

    這篇文章主要為大家詳細介紹了Golang中chan在接收和發(fā)送數(shù)據(jù)時因為“復制”而產(chǎn)生的開銷,文中的示例代碼講解詳細,感興趣的小伙伴可以了解下
    2024-01-01
  • Golang?gRPC?HTTP協(xié)議轉(zhuǎn)換示例

    Golang?gRPC?HTTP協(xié)議轉(zhuǎn)換示例

    這篇文章主要為大家介紹了Golang?gRPC?HTTP協(xié)議轉(zhuǎn)換示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-06-06
  • 基于Golang實現(xiàn)內(nèi)存數(shù)據(jù)庫的示例詳解

    基于Golang實現(xiàn)內(nèi)存數(shù)據(jù)庫的示例詳解

    這篇文章主要為大家詳細介紹了如何基于Golang實現(xiàn)內(nèi)存數(shù)據(jù)庫,文中的示例代碼講解詳細,具有一定的借鑒價值,需要的小伙伴可以參考一下
    2023-03-03
  • golang進行簡單權(quán)限認證的實現(xiàn)

    golang進行簡單權(quán)限認證的實現(xiàn)

    本文主要介紹了golang簡單權(quán)限認證的實現(xiàn),文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-09-09

最新評論