D 的个人博客

全职做开源,自由职业者

  menu

Go 边看边练 -《Go 学习笔记》系列(十)

上一篇: [1438596722873]


ToC


5.1 方法定义

方法总是绑定对象实例,并隐式将实例作为第一实参 (receiver)。

  • 只能为当前包内命名类型定义方法。
  • 参数 receiver 可任意命名。如方法中未曾使用,可省略参数名。
  • 参数 receiver 类型可以是 T*T。基类型 T 不能是接口或指针。
  • 不支持方法重载,receiver 只是参数签名的组成部分。
  • 可用实例 valuepointer 调用全部方法,编译器自动转换。

没有构造和析构方法,通常用简单工厂模式返回对象实例。

 1type Queue struct {
 2	elements []interface{}
 3}
 4
 5func NewQueue() *Queue { // 创建对象实例。
 6	return &Queue{make([]interface{}, 10)}
 7}
 8
 9func (*Queue) Push(e interface{}) error { // 省略 receiver 参数名。
10	panic("not implemented")
11}
12
13// func (Queue) Push(e int) error { // Error: method redeclared: Queue.Push
14// 		panic("not implemented")
15// }
16
17func (self *Queue) length() int { // receiver 参数名可以是 self、this 或其他。
18	return len(self.elements)
19}

方法不过是一种特殊的函数,只需将其还原,就知道 receiver T*T 的差别。

从 1.4 开始,不再支持多级指针查找方法成员。

 1type X struct{}
 2
 3func (*X) test() {
 4	println("X.test")
 5}
 6
 7func main() {
 8	p := &X{}
 9    p.test()
10    
11    // Error: calling method with receiver &p (type **X) requires explicit dereference
12    // (&p).test()
13}

5.2 匿名字段

可以像字段成员那样访问匿名字段方法,编译器负责查找。

通过匿名字段,可获得和继承类似的复用能力。依据编译器查找次序,只需在外层定义同名方法,就可以实现 "override"。

5.3 方法集

每个类型都有与之关联的方法集,这会影响到接口实现规则。

• 类型 T 方法集包含全部 receiver T 方法。
• 类型 *T 方法集包含全部 receiver T + *T 方法。
• 如类型 S 包含匿名字段 T,则 S 方法集包含 T 方法。
• 如类型 S 包含匿名字段 *T,则 S 方法集包含 T + *T 方法。
• 不管嵌入 T*T*S 方法集总是包含 T + *T 方法。

用实例 valuepointer 调用方法 (含匿名字段) 不受方法集约束,编译器总是查找全部方法,并自动转换 receiver 实参。

下一篇: [1438763483577]



社区小贴士

  • 关注标签 [golang] 可以方便查看 Go 相关帖子
  • 关注标签 [Go 学习笔记] 可以方便查看本系列
  • 关注作者后如有新帖将会收到通知

该文章同步自 黑客派