unsafe.Sizeof(T) 返回类型 T 值的内存大小,unsafe.Sizeof(*T) 返回指针本身大小(64 位系统为 8 字节);前者反映值布局,后者仅表示地址宽度,二者语义完全不同。

Golang中T和*T的unsafe.Sizeof大小_指针的大小是固定的

unsafe.Sizeof(T) 和 unsafe.Sizeof(*T) 的结果为什么不同

因为 unsafe.Sizeof 计算的是值本身的内存占用,不是类型定义或运行时对象开销。对任意类型 Tunsafe.Sizeof(T) 返回其值在内存中“平铺”所占字节数;而 unsafe.Sizeof(*T) 实际上是计算一个指向 T 的指针的大小——也就是机器字长:64 位系统上恒为 8,32 位系统上恒为 4

常见错误现象:
- 把 unsafe.Sizeof(*T) 当成 “T 类型实例加一层指针后的总开销”,误以为它等于 unsafe.Sizeof(T) + 8
- 在结构体嵌套指针字段时,用 unsafe.Sizeof 直接估算整个对象深度内存占用(它不递归)。

什么时候该用 unsafe.Sizeof(T),什么时候该用 unsafe.Sizeof(*T)

看你要测的对象是不是“已经存在的值”。T 是值类型或结构体字面量,*T 是一个指针变量(哪怕还没解引用)。它们对应完全不同的使用场景:

结构体里有 *T 字段时,unsafe.Sizeof 的结果包含什么

只包含指针字段本身(即 8 字节),绝不包含它指向的任何内容。这是最容易误解的一点:很多人以为 unsafe.Sizeof 会“穿透”指针,其实它连字段名都不认识,只按结构体布局硬算字节。

示例:

type S struct {
    A int64
    B *int64
}
s := S{A: 1, B: new(int64)}
fmt.Println(unsafe.Sizeof(s)) // 输出 16(64 位系统),不是 16 + 8

unsafe.Sizeof 的结果在不同 Go 版本或 GOARCH 下稳定吗

对同一 GOARCH,结果是稳定的;但跨架构或跨版本可能变——尤其是涉及结构体时。Go 不保证结构体字段布局,只保证导出字段的 ABI 兼容性(且仅限于相同 GOOS/GOARCH 下的同一 major 版本)。所以:

真正容易被忽略的是:哪怕你只跑在一个架构上,一旦结构体字段增减、重排、加 tag(比如 //go:notinheap),unsafe.Sizeof 就可能突变。它反映的是编译时快照,不是语义契约。

本文转载于:互联网 如有侵犯,请联系zhengruancom@outlook.com删除。
免责声明:正软商城发布此文仅为传递信息,不代表正软商城认同其观点或证实其描述。