通用的转换会发生内存拷贝,但是如下利用unsafe.Pointer实现的强转则不需要拷贝
func TestDemo(t *testing.T) {
a := "aaa"
b := "bbbbbbb"
// ssh 是 a 的内存地址
// 通过将字符串的底层地址强制转换成 StringHead 结构来获取字符串的底层指针和长度
// type StringHeader struct {
// Data uintptr // 指向字符串实际内容的指针
// Len int // 字符串的长度
// }
ssha := *(*reflect.StringHeader)(unsafe.Pointer(&a))
sshb := *(*reflect.StringHeader)(unsafe.Pointer(&b))
// 使用 unsafe.Pointer 将 ssh 转换为 []byte 类型的指针
// 然后再通过 * 运算符将其解引用,得到了一个字节切片
// type SliceHeader struct {
// Data uintptr // 指向切片元素(字节数组)实际内容的指针
// Len int // 切片的长度
// Cap int // 切片的容量
// }
b1 := *(*[]byte)(unsafe.Pointer(&ssha))
b2 := *(*[]byte)(unsafe.Pointer(&sshb))
fmt.Printf("%v", b1)
fmt.Printf("%v", b2)
}
变量会携带有一组校验数据,用来证明它的整个生命周期是否在运行时完全可知。
如果变量通过了这些校验,就可以在栈上分配。否则就说它逃逸了,必须在堆上分配。
能引起变量逃逸到堆上的典型情况: