1.方法语法格式
Golang 中同时有函数和方法。
方法是一个包含了接收者(receiver)的函数,receiver 可以是内置类型或者自定义类型 struct 的一个值或者是一个指针。所有给定类型的方法组成该类型的方法集。
方法的定义语法格式如下:
func (r ReceiverType) funcName(parameters) (return values){
//body
}
如下面的这个例子,定义了一个新类型 Integer,它和 int 一样,只是为它增加了一个新方法 Less()。
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")
}
}
可以看出,Golang 在自定义类型的对象中没有 C++/Java 那种隐藏的 this 指针,而是在定义成员方法时显式声明了其所属的对象。
2.方法接收者为值与指针的区别
方法接收者为对象的指针与值有什么区别呢?
如果方法接收者为对象指针,则会修改原对象;如果方法接收者为对象值,那么在方法中被操作的是原对象的副本,不会影响原对象。
package main
import "fmt"
type Integer int
// double 乘 2
func (p *Integer) double() int {
*p = *p * 2
fmt.Printf("double p = %d\n", *p)
return 0
}
// square 平方
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)
}
输出结果:
double p = 4
i = 4
square p = 16
i = 4
3.小结
(1)方法接收者是值,对象的值和指针均可以调用该方法。
(2)方法接收者是指针,只能通过指针或 addressable values 调用。
type T struct {
a int
}
func (tv T) Mv(a int) int { return 0 } // value receiver
func (tp *T) Mp(f float32) float32 { return 1 } // pointer receiver
var t T
var pt *T
func makeT() T
f := t.Mv; f(7) // like t.Mv(7)
f := pt.Mv; f(7) // like (*pt).Mv(7)
f := pt.Mp; f(7) // like pt.Mp(7)
f := t.Mp; f(7) // like (&t).Mp(7)
f := makeT().Mp // invalid: result of makeT() is not addressable
(3)当方法的接收者是值时,不管是值调用还是指针调用,方法内部都是对原对象的副本进行操作,不会影响原对象。
(4)当方法的接收者是指针时,不管是值调用还是指针调用,方法内部都是通过指针对原对象进行操作,会影响原对象。
一句话,是否修改对象看方法的接收者是值(不修改)还是指针(修改)。
参考文献
GoLang之方法与接口
The Go Programming Language Specification: Method values