以一个例子来说明:
//misc.go
package misc
type S struct {
i int32
k int8
j int64
}
//main.go
func main() {
s := new(misc.S)
// 通过 unsafe Pointer 给未导出的字段赋值
// 因为字节对齐, 内存中实际分配
//int32|int8| int64 |
//|xxxx|x—|xxxxxxxx|
p := (*int32)(unsafe.Pointer(s))
*p = int32(10)
// 1.unsafe Pointer 和 uintptr 可以互相转换
// 2.unsafe Pointer 和其他类型的指针可以互相转换
// 3.unsafe Pointer 不能进行指针运算而 uintptr 能
p1 := (*int8)(unsafe.Pointer(uintptr(unsafe.Pointer(s)) + unsafe.Alignof(int32(0))))// 偏移 4
*p1 = int8(-128)
p2 := (*int64)(unsafe.Pointer(uintptr(unsafe.Pointer(s)) + unsafe.Alignof(s)))// 偏移 8
*p2 = int64(128)
fmt.Println(s)//&{10 -128 128}
fmt.Println(unsafe.Sizeof(*s))//16 因为字节对齐的缘故不是 13
fmt.Println(unsafe.Sizeof(s))//8 s 是指针,在 64 位系统里指针占 8 个字节,指针就是内存地址,大小跟系统有关跟语言无关
fmt.Println(unsafe.Alignof(s))//8 Alignof 返回该类型的对齐值,struct 的对齐值是成员类型对齐值的最大值
}