如果一个函数有接收者,那么这个函数就叫方法。
Go 语言里有两种类型的接收者:值接收者和指针接收者。如下:
// user 在程序里定义一个用户类型
type user struct {
name string
email string
}
// notify 使用值接收者实现了一个方法
func (u user) notify() {
fmt.Printf("Sending User Email To %s<%s>\n",
u.name,
u.email)
}
// changeEmail 使用指针接收者实现了一个方法
func (u *user) changeEmail(email string) {
u.email = email
}
有两个概念:调用者和接收者,这两者都可以是值和指针。
1 使用值接收者声明方法,调用时会使用这个值的一个副本来执行:
1.1 使用值来调用:
bill := user{"Bill", "bill@email.com"}
bill.notify()
notify方法会接收到一个bill的副本,然后进行操作。
1.2 使用指针来调用:
lisa := &user{"Lisa", "lisa@email.com"}
lisa.notify()
notify 操作的是一个副本,只不过这次操作的是从 lisa 指针指向的值的副本。Go 编译器为了支持这种方法调用背后做的事情就 是将指针解引用为值。实际上执行的类似于:
(*lisa).notify()
2 使用指针接收者,方法会共享调用方法时接收者所指向的值:
2.1 使用指针来调用:
lisa := &user{"Lisa", "lisa@email.com"}
// 指向 user 类型值的指针可以用来调用
// 使用指针接收者声明的方法
lisa.changeEmail("lisa@newdomain.com")
一旦 changeEmail 调用返回,这个调用对值做的修改也会反映在 lisa指针所指向的值上。这是因为 changeEmail 方法使用了指 针接收者。
2.3 使用值来调用:
bill := user{"Bill", "bill@email.com"}
// user 类型的值可以用来调用
// 使用指针接收者声明的方法
bill.changeEmail("bill@newdomain.com")
Go 语言再一次对值做了调整,使之符合函数的接收者,进行调用,也就是先引用 bill 值得到一个指针,这样这个指针就能够 匹 配方法的接收者类型,再进行调用。实际上执行的类似于:
(&bill).changeEmail ("bill@newdomain.com")
总结一下,值接收者使用值的副本来调用方法,而指针接受者使用实际值来调用方法。具体需要用什么来作为使用者就看你的功能里面需要什么来做选择了。