大家都知道所有的指针变量都是一个内存位置,每个内存位置都有其定义的地址,可以使用&运算符来访问它,这个运算符表示内存中的地址。
func main() { a := 10 fmt.Printf("%x\n", &a) //打印变量a的地址 返回:c0000140a8 }
指针概念:指针是一个变量,其值是另一个变量的地址,即存储器位置的直接地址。
Go语言使用 * 运算符来取得指向指针存储的内存地址所对应的值(指针的格式化为%p),Go语言指针不支持运算,也不支持 -> 运算符,可以直接用,访问目标成员(有涉及到struct以后再写)
func main() { a := 10 ip := &a value := *ip fmt.Printf("a的地址:%x\n", &a) fmt.Printf("ip的地址:%x\n", &ip) fmt.Printf("ip的值:%d\n", *ip) fmt.Printf("value的值:%d", value) } 结果: a的地址:c0000140a8 ip的地址:c000006028 ip的值:10 value的值:10
nil指针
当一个指针被定义后没有分配到任何变量时,它的默认值为 nil。指针变量通常缩写为 ptr。nil指针是在几个标准库中定义的值为零的常量。
注意:对一个空指针的方向引用是不合法的,并会使程序奔溃。(var p *int = nil; *p = 10)
func main() { var ptr *int fmt.Printf("prt的值是:%x\n", ptr) if ptr != nil { fmt.Println("非空") } else { fmt.Println("空") } } prt的值是:0 空
指针的指针
func main() { var a *int //定义一个a指针,指向nil aP := &a //空指针a的指针 fmt.Printf("a-->nil:%x\n", a) fmt.Printf("a指针的内存地址:%x\n", &a) fmt.Printf("aP-->a:%x\n", aP) fmt.Printf("aP-->a-->nil(指针aP指向的指针a的内存地址):%x\n", *aP) } 结果: a-->nil:0 a指针的内存地址:c000006028 aP-->a:c000006028 aP-->a-->nil(指针aP指向的指针a的内存地址):0
如果定义一个指针的指针,可以这样定义:var ptr **int;
func main() { a := 10 aP := &a aPP := &aP fmt.Printf("a:%d\n", a) fmt.Printf("aP:%x\n", aP) fmt.Printf("*aP:%d\n", *aP) fmt.Printf("aPP:%x\n", aPP) fmt.Printf("*aPP:%x\n", *aPP) fmt.Printf("**aPP:%d\n", **aPP) } 结果: a:10 aP:c0000140a8 //ap是a的内存地址 *aP:10 aPP:c000006028 //app是ap的内存地址 *aPP:c0000140a8 //这里的是ap的地址 **aPP:10 //这才能拿到a的值
提醒各位一句:禁止套娃!!!
指针数组
指向数组的指针也会有多个值,确切说是指针数组有多个值
const MAX int = 3 //使用三个整数,将他们存储在指针数组中 func main() { a := []int{10, 100, 200} //初始化a数组 var ptr [MAX]*int //这里将ptr声明为一个指针数组,数组长度为MAX,ptr中的每个元素现在保存一个指向int值的指针 for i := 0; i < MAX; i++ { ptr[i] = &a[i] fmt.Printf("a[%d]的地址:%x\n", i, ptr[i]) } for i := 0; i < MAX; i++ { fmt.Printf("a[%d]的值是:%d\n", i, *ptr[i]) } } 结果: a[0]的地址:c00007e120 a[1]的地址:c00007e128 a[2]的地址:c00007e130 a[0]的值是:10 a[1]的值是:100 a[2]的值是:200
指针传递给函数
func main() { a := 100 b := 200 fmt.Printf("交换之前a的值为:%d\n", a) fmt.Printf("交换之前b的值为:%d\n", b) //把a和b的地址传递给函数,然后a和b的地址交换,最终实现两个值交换 swap(&a, &b) fmt.Printf("交换之后的a值为:%d\n", a) fmt.Printf("交换之后的b值为:%d\n", b) } func swap(x *int, y *int) { fmt.Printf("原a的地址:%x\n", &x) fmt.Printf("原b的地址:%x\n", &y) fmt.Printf("原a:%d\n", *x) fmt.Printf("原b:%d\n", *y) var temp int temp = *x *x = *y *y = temp } 结果: 交换之前a的值为:100 交换之前b的值为:200 原a的地址:c00009e020 原b的地址:c00009e028 原a:100 原b:200 交换之后的a值为:200 交换之后的b值为:100
作者:Fighting社火底子