php小編新一在使用Go語言進行http post請求時,發現高內存使用率的問題。這個問題引發了他的思考,為什么會出現這樣的情況?經過調研和分析,他找到了一些可能的原因,并提出了一些解決方案。在本文中,我們將深入探討這個問題并給出解答。
問題內容
我有一個 go
應用程序在 k8s 容器內運行。它作為一個 rest api 工作,接收請求并將它們的請求寫入 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) ... }) }
登錄后復制
在上面的代碼中,它監聽路徑 /logs
并調用 http
將數據保存到 elasticsearch 中。當我使用下面的函數打印內存使用情況時,我可以看到 alloc
不斷增加,直到內存耗盡。如果我刪除 http.post
調用,內存使用量始終為 1 到 3mb。內存使用量不斷增加的原因可能是什么?
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) }
登錄后復制
解決方法
http 文檔多次提到:
客戶端完成后必須關閉響應正文:
這是文檔中的示例:
resp, err := http.Get("http://example.com/") if err != nil { // handle error } defer resp.Body.Close() body, err := io.ReadAll(resp.Body) // ...
登錄后復制
如果你不這樣做,就會發生泄漏,因為主體將永遠保留在內存中。