go unsafe package 的学习

Go语言指针操作
本文介绍了Go语言中指针的基本类型及其应用场景,包括如何利用unsafe包进行指针运算及类型转换,并通过示例代码展示了如何获取结构体字段的内存地址。

go 不像c 那样直接可以对指针做一些操作,所以go 提供了一个标准的package 来操作指针。go 有三种指针类型:
1.  *  普通指针, 用于传递对象地址,不能进行指针运算

2.  unsafe.Pointer:通用指针类型,用于转换不同类型的指针,不能进行指针运算。

3.  uintptr:用于指针运算,GC 不把 uintptr 当指针,uintptr 无法持有对象。uintptr 类型的目标会被回收


unsafe.Pointer 可以和 普通指针 进行相互转换, unsafe.Pointer 可以和 uintptr 进行相互转换; 也就是说 unsafe.Pointer 是桥梁,可以让任意类型的指针实现相互转换,也可以将任意类型的指针转换为 uintptr 进行指针运算

我们可以看下面的code: 

package main

import (
	"fmt"
	"unsafe"
)

func main() {
	s := struct {
		a byte
		b byte
		c byte
		d int64
	}{0, 0, 0, 0}
	p := unsafe.Pointer(&s)
	up0 := uintptr(p)
	pb := (*byte)(p)
	*pb = 10
	fmt.Println(s)     // 10, 0, 0 , 0

	up := up0 + unsafe.Offsetof(s.b)
	p = unsafe.Pointer(up)
	pb = (*byte)(p)
	*pb = 20
	fmt.Println(s)   //10, 20, 0, 0

	up = up0 + unsafe.Offsetof(s.c)
	p = unsafe.Pointer(up)
	pb = (*byte)(p)
	*pb = 30
	fmt.Println(s)   // 10, 20, 30. 0

	up = up0 + unsafe.Offsetof(s.d)
	p = unsafe.Pointer(up)
	pi := (*int64)(p)
	*pi = 40
	fmt.Println(s)    //10, 20, 30, 40 
}



每种类型都有它的大小和对齐值,可以通过 unsafe.Sizeof 获取其大小,通过 unsafe.Alignof 获取其对齐值,通过 unsafe.Offsetof 获取其偏移量。不过 unsafe.Alignof 获取到的对齐值只是该类型单独使用时的对齐值,不是作为结构体字段时与其它对象间的对齐值,这里用不上,所以需要用 unsafe.Offsetof 来获取字段的偏移量,进而确定其内存地址



<think>好的,用户想了解Golang中unsafe.Sizeof的用法和引入目的。首先,我需要回忆一下这个函数的基本信息。根据引用[4],unsafe.Sizeof返回的是类型的大小,以字节为单位,但不会计算可能引用的内存,比如切片的话,返回的是描述符的大小,而不是底层数组的大小。 用户可能需要知道这个函数的使用场景,比如在需要精确控制内存布局或者进行底层操作时。比如在处理结构体内存对齐,或者与C语言交互的时候,知道每个类型的大小很重要。引用[2]和[3]中的例子显示,切片的大小是24字节,这可能是因为切片的结构包含指针、长度和容量。 然后要解释引入的目的。根据引用[4]和[5],这个函数属于unsafe包,所以主要用于低级别的编程,比如优化内存使用或与外部系统交互时。比如结构体的字段顺序不同可能导致大小不同,使用unsafe.Sizeof可以查看每个字段的大小,帮助调整结构体布局减少内存占用。 接下来要考虑用户可能的疑问,比如为什么字符串的Sizeof返回的是16字节(在64位系统)。根据引用[5],字符串结构体包含Data和Len两个字段,各占8字节和int的大小,所以总共是16字节。需要说明这点,避免用户误解为字符串内容的大小。 还要提醒用户注意使用unsafe包的注意事项,比如不同平台的结果可能不同,比如int的大小取决于系统位数。此外,频繁使用unsafe可能导致代码不可移植或存在安全风险,需要谨慎使用。 最后,生成相关问题时要覆盖常见的使用场景、注意事项、与其他函数的对比,以及实际应用例子。比如性能优化、结构体内存对齐、与C交互的例子等,帮助用户进一步探索。</think>### Golang中unsafe.Sizeof的用法及引入目的 #### 一、基本用法 `unsafe.Sizeof` 是Go语言 `unsafe` 包提供的函数,用于**获取变量或类型的底层内存占用大小(以字节为单位)**。其语法为: ```go unsafe.Sizeof(x) // 返回uintptr类型的值 ``` **特点**: 1. **类型描述符大小**:对于引用类型(如切片、字符串、映射),返回的是其**描述符**的大小,而非实际数据占用的内存[^2][^4]。 2. **编译期计算**:结果在编译时确定,无需运行时分配内存[^4]。 3. **内存对齐影响**:结果包含因内存对齐产生的填充字节[^4]。 #### 二、代码示例 ```go package main import ( "fmt" "unsafe" ) type Data struct { a int32 // 4字节 b bool // 1字节(可能填充为4字节) } func main() { var s string var slice []int data := Data{} fmt.Println("string类型大小:", unsafe.Sizeof(s)) // 输出16(64位系统) fmt.Println("切片描述符大小:", unsafe.Sizeof(slice)) // 输出24 fmt.Println("结构体Data大小:", unsafe.Sizeof(data)) // 输出8(含对齐填充) } ``` #### 三、核心应用场景 1. **内存布局优化** 通过分析结构体字段的内存对齐情况,调整字段顺序以减少填充空间: ```go type A struct { // 总大小12字节 a bool // 1字节 + 3填充 b int32 // 4字节 c bool // 1字节 + 3填充 } type B struct { // 总大小8字节 a bool // 1字节 c bool // 1字节 + 2填充 b int32 // 4字节 } ``` 2. **C语言交互** 在调用C函数时,需确保Go类型与C结构体大小一致: ```go /* #include <stdint.h> struct CData { int32_t num; char flag; }; */ import "C" type GoData struct { num int32 flag byte } func validate() { if unsafe.Sizeof(GoData{}) != C.sizeof_struct_CData { panic("内存布局不匹配") } } ``` 3. **底层系统编程** 例如实现自定义内存池时,精确计算对象内存需求。 #### 四、引入目的 1. **提供底层内存控制能力** 允许开发者绕过Go的类型安全限制,直接操作内存,用于性能关键场景[^4]。 2. **兼容性需求** 在与硬件交互或实现系统级功能(如操作系统内核、驱动开发)时,需精确控制内存布局[^1]。 3. **性能分析工具开发** 帮助开发者分析程序内存占用,例如实现内存分析器时统计对象大小。 #### 五、注意事项 1. **平台依赖性** `int` 类型大小随系统架构变化(32位系统4字节,64位系统8字节)。 2. **非安全操作** `unsafe` 包可能引发内存错误,需通过 `go vet` 静态检查: ```bash go vet -unsafeptr ./... ``` 3. **替代方案优先** 常规开发应优先使用 `reflect.Type.Size()`,仅在必要时使用 `unsafe.Sizeof`。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值