golang程序性能提升改进篇之文件的读写---第一篇
创始人
2025-01-07 15:33:09
0

背景:接手的项目是golang开发的(本人初次接触golang)经常出现oom。这个程序是计算和io密集型,调用流量属于明显有波峰波谷,但是因为各种原因,当前无法快速通过serverless或者动态在高峰时段调整资源,低峰释放资源的方式进行解决,因此只能通过分析内存和cpu消耗热点,然后改动代码,进而解决服务的稳定性和性能问题。

思路
通过搜索了解golang程序如何进行性能分析,当时就是使用pprof了,通过分析发现了程序有一些使用内存较多的地方,例如我们读取大量文件,通过分析发现io.ioutil.ReadAll消耗内存较多,通过查询发现io.copy可以减少一些内存消耗。

在这里插入图片描述

同时也在网上发现其他开源项目与我们遇到的情况类似。 https://github.com/elastic/beats/issues/36151是某个开源软件中将ioutil.ReadAll 更新为io.Copy的讨论,里面也有他们的对比测试 。

性能对比测试:

在本机上写了一段代码,ioutil.ReadAll和io.Copy的go benchmarktest结果如下
在这里插入图片描述

代码比较简单 main.go

package main  import ( 	"bytes" 	"fmt" 	"io" 	"io/ioutil" 	"net/http" 	"time" )  func fetchWithReadAll(url string) ([]byte, time.Duration, error) { 	// 发起HTTP GET请求 	resp, err := http.Get(url) 	if err != nil { 		return nil, 0, err 	} 	defer resp.Body.Close()  	// 使用 io.ReadAll 读取响应内容 	start := time.Now() 	data, err := ioutil.ReadAll(resp.Body) 	if err != nil { 		return nil, 0, err 	} 	duration := time.Since(start) 	return data, duration, nil }  func fetchWithCopy(url string) ([]byte, time.Duration, error) { 	// 发起HTTP GET请求 	resp, err := http.Get(url) 	if err != nil { 		return nil, 0, err 	} 	defer resp.Body.Close()  	// 使用 io.Copy 读取响应内容 	start := time.Now() 	var buffer bytes.Buffer 	_, err = io.Copy(&buffer, resp.Body) 	if err != nil { 		return nil, 0, err 	} 	duration := time.Since(start) 	return buffer.Bytes(), duration, nil }  func main() {     //  此url能下载一个大约1-2mb的文本文件 	url := "http://x.y.z/abc.txt"  	// 测试 ioutil.ReadAll 	dataReadAll, durationReadAll, err := fetchWithReadAll(url) 	if err != nil { 		fmt.Printf("Error fetching with ReadAll: %v\n", err) 		return 	} 	fmt.Printf("ReadAll took %v for %d bytes\n", durationReadAll, len(dataReadAll))  	// 测试 io.Copy 	dataCopy, durationCopy, err := fetchWithCopy(url) 	if err != nil { 		fmt.Printf("Error fetching with Copy: %v\n", err) 		return 	} 	fmt.Printf("Copy took %v for %d bytes\n", durationCopy, len(dataCopy)) } 

main_test.go

package main  import ( 	"testing" )  // 假设我们已有 fetchWithReadAll 和 fetchWithCopy 函数定义  // BenchmarkFetchWithReadAll 为 fetchWithReadAll 函数编写基准测试。 func BenchmarkFetchWithReadAll(b *testing.B) { 	//  此url能下载一个大约1-2mb的文本文件 	url := "http://x.y.z/abc.txt 	b.ReportAllocs() 	for i := 0; i < b.N; i++ { 		_, _, err := fetchWithReadAll(url) 		if err != nil { 			b.Error(err) 		} 	} }  // BenchmarkFetchWithCopy 为 fetchWithCopy 函数编写基准测试。 func BenchmarkFetchWithCopy(b *testing.B) { 	//  此url能下载一个大约1-2mb的文本文件 	url := "http://x.y.z/abc.txt" 	b.ReportAllocs() 	for i := 0; i < b.N; i++ { 		_, _, err := fetchWithCopy(url) 		if err != nil { 			b.Error(err) 		} 	} }  

最后执行
···
go test -bench=.

···

相关内容

热门资讯

教程辅助“潮汕掌上娱脚本”开挂... 教程辅助“潮汕掌上娱脚本”开挂(透视)辅助安装-知乎;无需打开直接搜索加(薇:136704302)咨...
八分钟辅助“潮汕来物局有透视软... 八分钟辅助“潮汕来物局有透视软件吗”开挂(透视)辅助插件存在挂教程-哔哩哔哩1、下载安装好潮汕来物局...
技术分享“陕麻圈延安划水辅助”... 陕麻圈延安划水辅助是一款可以让一直输的玩家,快速成为一个“必胜”的ai辅助神器,有需要的用户可以加我...
教程辅助“雀神挂件价格开挂透视... 教程辅助“雀神挂件价格开挂透视”开挂(透视)辅助安装-知乎;亲,雀神挂件价格开挂透视这款游戏原来确实...
四分钟辅助“潮汕汇软件辅助”开... 四分钟辅助“潮汕汇软件辅助”开挂(透视)辅助软件科技教程-哔哩哔哩 了解更多开挂安装加(136704...
盘点十款“钱塘十三水辅助器软件... 盘点十款“钱塘十三水辅助器软件”附开挂工具辅助详细教程;亲,钱塘十三水辅助器软件这款游戏原来确实可以...
教程辅助“新玄龙辅助工具”开挂... 教程辅助“新玄龙辅助工具”开挂(透视)辅助平台-哔哩哔哩 了解更多开挂安装加(136704302)微...
详细辅助“小逸碰胡插件脚本”开... 详细辅助“小逸碰胡插件脚本”开挂(透视)辅助器普及教程-知乎【无需打开直接搜索加薇136704302...
科技通报“南通长牌辅助”附开挂... 科技通报“南通长牌辅助”附开挂安装辅助详细教程;无需打开直接搜索打开薇:136704302 咨询了解...
教程辅助“辅助挂定制交易平台”... 教程辅助“辅助挂定制交易平台”开挂(透视)辅助下载-知乎;打开点击测试直接进入微信(13670430...