Golang 程序中定义的类型有与其相关的方法。让我们来看一个例子:
```go
type T struct {
name string
}
func (t T) PrintName() {
fmt.Println(t.name)
}
func main() {
t := T{name: "Michał"}
t.PrintName()
}
```
正如您可能怀疑程序输出名字。方法是一个函数,拥有附加的,单独元素的参数列表,称之为接收器。它被放在一个方法名之前。接收器的类型决定了如何使用方法。
## 接收器
方法被绑定到接收器的基类。小例子解释的很好:
```go
type T struct {
name string
}
func (T) F() {}
func (*T) F() {}
```
上面的代码无法编译. 第一个 F 方法被绑到 T 上. 第二各方法被邦到 *T 上. 对于单个基类型,方法名必须唯一,所以编译将报错:
```
> go install github.com/mlowicki/lab && ./spec/bin/lab
# github.com/mlowicki/lab
spec/src/github.com/mlowicki/lab/lab.go:103: method redeclared: T.F
method(T) func()
method(*T) func()
```
如果基类是一个 struct 类型,那么字段名不能和方法名重复。
```go
type T struct {
M int
}
func (t T) M() {}
```
编译失败信息如下:
```
type T has both field and method named M
```
Go 的类型系统限制了哪些类型可以作为接收器. 它不能是接口类型或指针,因此它不可能为一个空接口(interface{})定义方法来满足所有类型。 只允许使用类型名称,所以,类型字面值会导致编译错误:
```go
func (v map[string]float64) M() {}
```
错误信息 `invalid receiver type map[string]float64 (map[string]float64 is an unnamed type).`
## 方法集
调用一个 T 类型的变量上的 m 方法, 方法 m 必须在与 T 关联的方法集上. 方法集中的一个成员是什么意思?
### 接口类型
接口类型的方法集是接口自己 — 为实现接口需要定义方法列表:
```go
type I interface {
F()
G()
}
type T struct{}
func (T) F() {}
func (T) G() {}
func (T) H() {}
func main() {
var i I = T{}
i.F()
i.G()
i.H() // error: i.H undefined (type I has no field or method H)
}
```
上面的例子中,不允许调用 T 类型的变量 i 上的方法。仅有来自接口的接口类型值方法属于接口类型的方法集。(方法集是静态的,当分配不同类型的值时不会改变)。 要调用 H 方法,需要先使用类型断言:
```go
t, ok := i.(T)
if !ok {
log.Fatal("Type assertion failed")
}
t.H()
```
### 非接口类型
对于非接口类型 T,方法集由接收器为 T 的方法组成:
```go
type T struct{}
func (T) F() {}
func (T) G() {}
type U struct{}
func (U) H() {}
func main() {
t := T{}
t.F()
t.G()
t.H() // error: t.H undefined (type T has no field or method H)
}
```
当处理 T 类型的指针时, 它的方法集由接收器为 T 或 *T 的方法组成:
```go
type T struct{}
func (T) F() {}
func (T) G() {}
func (*T) H() {}
func main() {
t := &T{}
t.F()
t.G()
t.H()
}
```
为什么 T 的方法集没有 *T 类型接收器的方法,而 *T 的方法集却包含 T 类型接收器的方法呢? 令人惊讶的是,分配给 t 不是地址,但无指针结构值仍执行的很好:
```go
t := T{}
t.F()
t.G()
t.H()
```
Go 规范描述了明确的情况:
> 如果 x 是可寻址的,并且 x 的方法集包含 m ,x.M() 是 (&x).M() 的速记。
何时使用值还是指针接收器的建议放在了官方的 [FAQ](https://golang.org/doc/faq#methods_on_values_or_pointers)中。
### 唯一性
类型 T 定义的方法集不能有俩个名字一样的方法。 有相同的方法名但不同类型的参数,在 Go 中是不可能的。 ( Go 中没有 [ad hoc polymorphism](https://en.wikipedia.org/wiki/Ad_hoc_polymorphism) )。
> 类型 T 定义的方法集由 T 实现.
## 打印方法集
Go 有 [reflect](https://golang.org/pkg/reflect/) 包,对于看类型的方法集很有用:
```go
func PrintMethodSet(val interface{}) {
t := reflect.TypeOf(val)
fmt.Printf("Number of methods: %d\n", t.NumMethod())
for i := 0; i < t.NumMethod(); i++ {
m := t.Method(i)
fmt.Printf("Method %s\n", m)
fmt.Printf("\tName: %s\n", m.Name)
fmt.Printf("\tPackage path: %s\n", m.PkgPath)
}
}
```
从 [Method](https://golang.org/pkg/reflect/#Value.Method) 返回值类型,方法是 [Method](https://golang.org/pkg/reflect/#Method) 类型.
值得一提的是,文档有个 bug, 因为 [NumMethod](https://golang.org/pkg/reflect/#Value.NumMethod) 从方法集返回了方法的数量,但只有那些被导出的方法 (名字以大写字母开头)。 归档在 [#17686](https://github.com/golang/go/issues/17686).
## 引入类型方法
方法只能被定义在有类型创建的包里。
github.com/mlowicki/lib/lib.go:
```go
package lib
type T struct{}
```
github.com/mlowicki/lab/lab.go:
```go
package main
import (
"fmt"
. "github.com/mlowicki/lib"
)
func (T) F() {}
func (*T) F() {}
[...]
```
这会引起构建错误 `cannot define new methods on non-local type lib.T`.
## 不使用参数
函数/方法定义不强制命名所有参数或接收者。如果不使用它们,则只能指定类型。 Name can be eventually introduced later on when it’ll actually needed:
```go
type T struct {
name string
}
func (t T) F(name string) {}
func (T) G(string) {}
```
G 的声明省略无用标识符。
本文由 GCTT 原创编译,Go语言中文网 荣誉推出
有疑问加站长微信联系(非本文作者))