定义
指针,就是保存对象的内存地址。用指针的好处就是节约内存空间。
定义一个指针类型的变量很简单,就是在变量名前加*
//基础类型指针
var ps *string
var pi *int
var pf *float64
var pb *bool
//结构体指针
type Person struct {
name string
age int
}
var pperson *Person
使用
在把指针使用的得心应手之前,首先要熟悉go语言中这两个操作符:*与&。
先说说&。&变量名,表示对变量取地址。&可以理解为一个方法简写,这个方法的唯一参数是变量名,返回值是一个内存地址(即一个指针类型变量)。&的使用如下:
//先定义指针
var ps *string
//再定义变量
var s string
s = "i am string"
//通过&操作符取s的内存地址,地址保存到指针ps中
ps = &s
fmt.Printf("value of s:%s\n",s)
fmt.Printf("value of ps:%v\n",ps)
以上代码的执行结果:
value of s:i am string
value of ps:0xc42007a1c0
需要注意的一点就是指针与变量必须是同类型的。你不能把一个int类型的变量地址赋给一个string类型的指针。
再来说说*。*在指针的使用中有两个角色:
其一,作为指针类型的定义符,本文开篇就用到了
其二,与&符做相反操作,&变量名是对变量取地址,而*指针名则是对指针地址取其保存的内容。同样可以理解*是一个方法的简写,此方法唯一参数是一个指针类型变量(也就是一个内存地址),返回值是这个地址保存的变量真实值。接上段代码:
fmt.Printf("ps point to:%s\n",*ps)
输出:
ps point to:i am string
从这三行输出上可以很明确的看出两个操作符的作用了。需要注意的一点就是有两个用途,在看别人的代码时,把这两种场景区分开,明确了解每个起的什么作用,就不会迷失于*&之海了。
上面只是演示指针怎么用,而实际项目当中最常用的场景就是用指针作方法返回值和参数了。不难理解,在函数传参和返回值的时候,用指针类型代替值类型,通过引用取值(指针就是对值的引用),复用已创建对象,减少内存开销,当然就达到了节约内存,提高程序效率的目的。实际传递的数据类型往往是结构体:
//结构体指针
type Person struct {
Name string
Age int
}
var pperson *Person
var person Person
person = Person{
"luna",
20,
}
pperson = &person
然后定义一个方法,对于传入的Person对象,年龄增加1并返回,参数和返回值都是用指针:
func yearGone(p *Person) *Person {
p.age += 1
return p
}
func main() {
person = Person{
"luna",
20,
}
fmt.Printf("last year:%v\n", person)
p := yearGone(&person)
fmt.Printf("now:%v\n", *p)
fmt.Printf("person now:%v\n", person)
}
输出:
last year:{luna 20}
now:{luna 21}
person now:{luna 21}
这里从头到尾,就只创建一个Person对象person,所有操作都是对它自己。注意打印p的内容时用了*p,因为p是指针类型变量,直接打印p只是个地址而已。
当你看别人的代码困惑与为什么这里用了指针而那里没用时,我的建议是永远不要问别人为什么用了(没用)指针-_-。有时这个原因只是“看心情而已”。