
本文介绍如何正确使用 go 的反射机制获取任意类型(尤其是结构体)的内存占用大小,指出常见误区,并提供简洁可靠的实现方案。
本文介绍如何正确使用 go 的反射机制获取任意类型(尤其是结构体)的内存占用大小,指出常见误区,并提供简洁可靠的实现方案。
在 Go 中,开发者有时需要获取某个值在内存中实际占用的字节数(即其“大小”),例如用于性能分析、序列化预估或底层内存布局调试。这与 C 语言中的 sizeof 操作符功能类似。但需特别注意:不能对 reflect.Value 调用 unsafe.Sizeof(),因为该调用返回的是 reflect.Value 结构体本身的固定开销(通常为 12 或 24 字节,取决于架构),而非其所包装目标类型的大小。
正确的做法是通过 reflect.TypeOf() 获取类型的 reflect.Type,再调用其 Size() 方法——该方法返回的是该类型在内存中实际对齐后的尺寸(单位:字节),已包含结构体字段间的填充(padding)。
以下是一个简洁、通用且类型安全的实现:
package main
import (
"fmt"
"reflect"
)
// SizeOf 返回任意值在内存中占用的字节数(含对齐填充)
func SizeOf[T any](v T) uintptr {
return reflect.TypeOf(v).Size()
}
func main() {
type MyStruct struct {
a int // 8 bytes (on amd64)
b int64 // 8 bytes
c float32 // 4 bytes
d float64 // 8 bytes
e float64 // 8 bytes
}
s := MyStruct{}
fmt.Printf("Size of MyStruct: %d bytes\n", SizeOf(s)) // 输出: 40
}✅ 关键说明:
- reflect.TypeOf(v).Size() 是唯一推荐的标准方式,它基于编译时确定的类型信息,准确反映运行时内存布局;
- 该方法适用于所有类型:基础类型(int, string)、复合类型(struct, array)、甚至指针和接口(但注意:接口本身大小 ≠ 其动态值大小);
- 对于结构体,结果包含字段对齐所需的填充字节(如上述示例中 float32 后有 4 字节填充,使后续 float64 满足 8 字节对齐);
⚠️ 注意事项:
- Size() 返回的是静态类型大小,不适用于切片、映射、通道等引用类型所指向的底层数据(例如 []int 的 Size() 仅返回 slice header 大小,约 24 字节);
- 若需获取动态分配数据的实际内存用量(如 slice 底层数组总长 × 元素大小),需额外计算,不可依赖 Size();
- 泛型约束 T any 确保函数可接受任意类型,且避免了运行时反射开销(相比 interface{} + reflect.ValueOf 更高效、更安全)。
综上,reflect.TypeOf(x).Size() 是 Go 中替代 C 风格 sizeof 的标准、可靠且零依赖的解决方案。










