03-05-10-Go黑科技之-指针运算

一 Go指针

我们大家都知道,go语言指针是不支持运算的,也就是不支持++/--操作,但是我们借助于unsafe包,可以完成这个操作

我们现在要定义一个数组a:=[3]int8{6,8,9}

取出a[0]的指针,自增后,解引用,即可得到a[1]对应的值 8

二 指针,uintptr,unsafe.Pointer

2.1 *类型

普通指针,用于传递对象地址,不能进行指针运算。

2.2 unsafe.Pointer

通用指针类型,用于转换不同类型的指针,不能进行指针运算。

2.3 uintptr

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

2.4 结论

unsafe.Pointer 可以和 普通指针 进行相互转换。

unsafe.Pointer 可以和 uintptr 进行相互转换。

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

//uintptr这个类型,在golang中,字节长度也是与int一致。通常Pointer不能参与运算,比如你要在某个指针地址上加上一个偏移量,Pointer是不能做这个运算的,那么谁可以呢?就是uintptr类型了,只要将Pointer类型转换成uintptr类型,做完加减法后,转换成Pointer,通过*操作解引用即可

三 代码操作

package main

import (
        "fmt"
        "unsafe"
)

func main() {
    //定义一个长度为3的int8类型数组
        a:=[3]int8{6,8,9}
        //取出数组第一个位置的地址
        a_first_point:=&a[0]
        a_first_unsafe_point:=unsafe.Pointer(a_first_point)
        fmt.Println("a[0]的地址为:",a_first_unsafe_point)
        //指针只能一个字节字节取,int8占一个字节,所以看到值只加了1
        fmt.Println("a[1]的地址为:",unsafe.Pointer(&a[1]))
        //把a_first_unsafe_point转成uintptr类型,就可以指针运算了
        a_uintptr_first_unsafe_point:=uintptr(a_first_unsafe_point)
        //指针+1 表示到了数组的第二个位置
        a_uintptr_first_unsafe_point++
        fmt.Println("a[0]位置指针自增1后,的指针位置:",a_uintptr_first_unsafe_point)
        //打印出来可以看到跟&a[1]的地址是一样的
        a_uintptr_second_unsafe_point:=unsafe.Pointer(a_uintptr_first_unsafe_point)
        fmt.Println("a[0]位置指针自增1后,的指针位置,转成unsafe_Pointer类型:",a_uintptr_second_unsafe_point)
        //将该指针转换成 *int8类型(因为它本身就是*int8类型)
        int8_point:=(*int8)(a_uintptr_second_unsafe_point)
        //解引用,得到指针对应的结果,就是数组的第二个值,8
        fmt.Println(*int8_point)

}