Golang 中同时有函数和方法。方法是一个包含了接收者(receiver)的函数,receiver可以是内置类型或者自定义类型struct的一个值或者是一个指针。所有给定类型的方法组成该类型的方法集。
方法的定义语法格式如下:
func (r ReceiverType) funcName(parameters) (return values){ //body }
如下面的这个例子,定义了一个新类型Integer,它和int一样,只是为它增加了个新方法Less()。
package main import "fmt" type Integer int func (a Integer) Less(b Integer) bool { return a < b } func main() { var a Integer = 1 if a.Less(2) { fmt.Println("less then 2") } }
./hello less then 2
可以看出,Golang 在自定义类型的对象中没有C++/Java那种隐藏的this指针,而是在定义成员方法时显式声明了其所属的对象。
方法接收者为对象的指针与值有什么区别呢?如果方法接收者为对象的指针,则会修改原对象,如果方法接收者为对象的值,那么在方法中被操作的是原对象的副本,不会影响原对象。示例如下:
package main import "fmt" type Integer int //乘2 func (p *Integer) double() int { *p = *p * 2 fmt.Printf("double p = %d\n", *p) return 0 } //平方 func (p Integer) square() int { p = p * p fmt.Printf("square p = %d\n", p) return 0 } func main() { var i Integer = 2 i.double() //receiver 为对象的指针,原对象被修改 fmt.Println("i = ", i) i.square() //receiver 为对象的值,原对象不会被修改 fmt.Println("i = ", i) }
root@ubuntu:~/go_learn/example.com/hello# ./hello double p = 4 i = 4 square p = 16 i = 4
使用方法时注意以下几点:
(1)不管方法的 receiver 是对象的值还是指针,对象的值和指针均可以调用该方法。即对象的值既可以调用 receiver 是值的方法,也可以调用 receiver 是指针的方法。对象的指针也是如此;
(2)当方法的接收者是值时,即使是指针调用,那么方法内部也是对原对象的副本进行操作,不会影响原对象;
(3)当方法的接收者是指针时,即使用值调用,那么方法内部也是通过指针对原对象进行操作,会影响原对象。