go 内存分配优化

函数传参使用结构体指针还是结构体?

  • 如果函数不需要修改原始数据,并且数据量较小,建议使用值传递
  • 如果函数需要修改原始数据,或者数据量较大,建议使用指针传递

大量创建复杂对象如何优化?

哪一种初始化方式更好?

go 复制代码
type MyStruct strcut{
    A int
    ...
 }
 
 // 需要构建一个 []*MyStruct
 
// 方式一
// 优点: 申请内存次数为 批量申请内存比多次申请内存效率高
// 缺点: 由于elem申请的是连续的一片内存,任何一个元素被引用,elem都无法释放
arr := make([]*MyStruct, size)
elem = make([]MyStruct, size)
for i := 0; i < size; i++ {
    m[i] = &elem[i]
}

// 方式二
// 缺点: 申请内存次数为 1 + size次
arr:=make([]*MyStruct, size)
for i := 0; i < size; i++ {
    arr[i] = &MyStruct{}
}

结构体体字段定义成基本类型还是指针类型?

go 复制代码
// 内存占用 4个字节
type MyStruct strcut{
    A int
 }
 
 // // 内存占用 8个字节
 type MyStructPoint strcut{
    A *int
 }
  • 指针类型内存占用比基本类型大,所以应该少用指针
  • GC会扫描所有的指针字段, 所以使用MyStructPoint方式定义容易加重GC扫描负担

常驻内存需要维护一个只读map[string]any,有没有优化手段?

由于 go底层字符串的实现会包含一个指针,大量的字符串持有会 加重GC扫描负担 优化思路: 将所有的字符串都保存在byte数组中

go 复制代码
type StringHeader struct {
    Data uintptr
    Len  int
}

cloudwego 中的实现

go 复制代码
// StrMap represents GC friendly readonly string map implementation.
// type V must NOT contain pointer for performance concern.
type StrMap[V any] struct {
	// `data` holds bytes of keys
	data []byte

	// `items` holds key meta
	items []mapItem[V]

	// max hashtable ~ 2 billions which means len(items) < the num as well.
	hashtable []int32 // using int32 for mem efficiency

	// for maphash
	seed maphash.Seed
}

type mapItem[V any] struct {
	off  int
	sz   uint32 // 4GB, big enough for key
	slot uint32
	v    V
}

如何看逃逸分析

在 Go 语言里,逃逸分析这一重要任务是由编译器负责完成的。它会在编译阶段就进行细致的分析,以此来确定变量究竟该分配到栈上,还是堆上,此过程对程序的正常运行和性能表现起着关键作用。

从性能优化的角度考虑,我们要尽可能地减少堆上的内存分配。因为在堆上进行内存分配和回收操作的开销相对较大,会涉及到垃圾回收机制的频繁介入,这会消耗更多的系统资源和时间。所以,合理利用逃逸分析的结果,将变量分配到合适的内存区域,能有效提升程序的整体性能。

可以通过以下编译选项展示变量逃逸的情况,帮助代码优化:

bash 复制代码
go build -gcflags=-m main.go
相关推荐
qq_2975746713 小时前
【实战教程】SpringBoot 实现多文件批量下载并打包为 ZIP 压缩包
java·spring boot·后端
马猴烧酒.18 小时前
【面试八股|Java集合】Java集合常考面试题详解
java·开发语言·python·面试·八股
闻哥21 小时前
从测试坏味道到优雅实践:打造高质量单元测试
java·面试·单元测试·log4j·springboot
计算机程序设计小李同学21 小时前
基于 Spring Boot + Vue 的龙虾专营店管理系统的设计与实现
java·spring boot·后端·spring·vue
Charlie_lll1 天前
力扣解题-[3379]转换数组
数据结构·后端·算法·leetcode
南风知我意9571 天前
【前端面试5】手写Function原型方法
前端·面试·职场和发展
VX:Fegn08951 天前
计算机毕业设计|基于springboot + vue云租车平台系统(源码+数据库+文档)
数据库·vue.js·spring boot·后端·课程设计
汤姆yu1 天前
2026基于springboot的在线招聘系统
java·spring boot·后端
计算机学姐1 天前
基于SpringBoot的校园社团管理系统
java·vue.js·spring boot·后端·spring·信息可视化·推荐算法
java1234_小锋1 天前
Java高频面试题:SpringBoot如何自定义Starter?
java·spring boot·面试