go之指针
什么是指针
指针是一种存储变量内存地址(Memory Address)的变量
如上图所示,变量b的值为156,而b的内存地址为0x1040a124。变量a存储了b的地址,我们就称a指向了b。
& 放在变量前,表示取该变量的地址
* 放在类型前,表示该类型的指针
* 放在变量(指针变量)前,表示解引用
指针定义
package main import "fmt" func main() { var i =10 //定义int类型变量 var p =&i // 取i的地址,赋值给变量 p fmt.Println(p) //0xc00000a098 }
指针变量的类型
package main import "fmt" func main() { var i uint=10 //定义int类型变量 var p *uint=&i //*int表示 int类型的指针 fmt.Println(p) //0xc00000a098 }
解引用
package main import "fmt" func main() { var i uint=10 //定义int类型变量 var p *uint=&i //*int表示 int类型的指针 fmt.Println(*p) //解引用 10 }
指针变量的指针
package main import "fmt" func main() { var i int=10 var p *int=&i var x **int=&p var y ***int=&x fmt.Println(p) //0xc00000a098 fmt.Println(x) //0xc000006028 fmt.Println(y) //0xc000006030 fmt.Println(*y) //0xc000006028 fmt.Println(**y) //0xc00000a098 fmt.Println(***y) //10 }
指针类型的零值
引用类型零值是nil
package main import "fmt" func main() { var p *string fmt.Println(p) //nil 必须要 初始化才能用--》PyObject * }
向函数传递指针参数
修改会影响原来的
package main import "fmt" func main() { var i int = 10 testP(&i) //函数参数传递都是copy传递 fmt.Println("--", i) //-- 11 } func testP(a *int) { fmt.Println(a) // 地址 0xc00000a098 fmt.Println(*a) //10 *a++ fmt.Println(*a) //11 }
不要向函数传递数组的指针,而应该使用切片
假如我们想要在函数内修改一个数组,并希望调用函数的地方也能得到修改后的数组,一种解决方案是把一个指向数组的指针传递给这个函数
package main import ( "fmt" ) func modify(arr *[3]int) { (*arr)[0] = 90 } func main() { a := [3]int{89, 90, 91} modify(&a) fmt.Println(a) }
在上面程序的第 13 行中,我们将数组的地址传递给了 modify
函数。在第 8 行,我们在 modify
函数里把 arr
解引用,并将 90
赋值给这个数组的第一个元素。程序会输出 [90 90 91]
。
a[x]
是 (*a)[x]
的简写形式,因此上面代码中的 (*arr)[0]
可以替换为 arr[0]
。下面我们用简写形式重写以上代码。
package main import ( "fmt" ) func modify(arr *[3]int) { arr[0] = 90 } func main() { a := [3]int{89, 90, 91} modify(&a) fmt.Println(a) }
该程序也会输出 [90 90 91]
。
这种方式向函数传递一个数组指针参数,并在函数内修改数组。尽管它是有效的,但却不是 Go 语言惯用的实现方式。我们最好使用切片来处理
接下来我们用切片来重写之前的代码。
package main import ( "fmt" ) func modify(sls []int) { sls[0] = 90 } func main() { a := [3]int{89, 90, 91} modify(a[:]) fmt.Println(a) }
在上面程序的第 13 行,我们将一个切片传递给了 modify
函数。在 modify
函数中,我们把切片的第一个元素修改为 90
。程序也会输出 [90 90 91]
。所以别再传递数组指针了,而是使用切片吧。上面的代码更加简洁,也更符合 Go 语言的习惯。
Go不支持指针运算
Go 并不支持其他语言(例如 C)中的指针运算。
package main func main() { b := [...]int{109, 110, 111} p := &b p++ }
上面的程序会抛出编译错误:main.go:6: invalid operation: p++ (non-numeric type *[3]int)
。