php小編新一在使用Go語(yǔ)言進(jìn)行http post請(qǐng)求時(shí),發(fā)現(xiàn)高內(nèi)存使用率的問(wèn)題。這個(gè)問(wèn)題引發(fā)了他的思考,為什么會(huì)出現(xiàn)這樣的情況?經(jīng)過(guò)調(diào)研和分析,他找到了一些可能的原因,并提出了一些解決方案。在本文中,我們將深入探討這個(gè)問(wèn)題并給出解答。
問(wèn)題內(nèi)容
我有一個(gè) go 應(yīng)用程序在 k8s 容器內(nèi)運(yùn)行。它作為一個(gè) rest api 工作,接收請(qǐng)求并將它們的請(qǐng)求寫入 elasticsearch。
我的代碼是:
var r = gin.default()
r.post("/logs", func(c *gin.context) {
fmt.println("receive log event")
printmemusage()
jsondata, err := ioutil.readall(c.request.body)
d := strings.newreader(jsondata)
http.post(fmt.sprintf("%s/_bulk", geteshost()), "application/json", d)
...
})
}
登錄后復(fù)制
在上面的代碼中,它監(jiān)聽(tīng)路徑 /logs 并調(diào)用 http 將數(shù)據(jù)保存到 elasticsearch 中。當(dāng)我使用下面的函數(shù)打印內(nèi)存使用情況時(shí),我可以看到 alloc 不斷增加,直到內(nèi)存耗盡。如果我刪除 http.post 調(diào)用,內(nèi)存使用量始終為 1 到 3mb。內(nèi)存使用量不斷增加的原因可能是什么?
func bToMb(b uint64) uint64 {
return b / 1024 / 1024
}
func PrintMemUsage() {
var m runtime.MemStats
runtime.ReadMemStats(&m)
// For info on each, see: https://golang.org/pkg/runtime/#MemStats
fmt.Printf("Alloc = %v MiB", bToMb(m.Alloc))
fmt.Printf("\tTotalAlloc = %v MiB", bToMb(m.TotalAlloc))
fmt.Printf("\tSys = %v MiB", bToMb(m.Sys))
fmt.Printf("\tNumGC = %v\n", m.NumGC)
}
登錄后復(fù)制
解決方法
http 文檔多次提到:
客戶端完成后必須關(guān)閉響應(yīng)正文:
這是文檔中的示例:
resp, err := http.Get("http://example.com/")
if err != nil {
// handle error
}
defer resp.Body.Close()
body, err := io.ReadAll(resp.Body)
// ...
登錄后復(fù)制
如果你不這樣做,就會(huì)發(fā)生泄漏,因?yàn)橹黧w將永遠(yuǎn)保留在內(nèi)存中。






