分享
  1. 首页
  2. 文章

golang中接口赋值与方法集

StormZhu · · 1655 次点击 · · 开始浏览
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

接口使用疑问

golang中的接口可以轻松实现C++中的多态,而且没有继承自同一父类的限制,感觉方便很多。但是在使用的时候,如果没有理解,也可能会遇到"坑"。比如《Go语言实战》中的一个例子:

package main
import "fmt"
type user struct {
 name string
 email string
}
type notifier interface {
 notify()
}
func (u *user) notify() {
 fmt.Printf("sending user email to %s<%s>\n",
 u.name,
 u.email)
}
func sendNotification(n notifier) {
 n.notify()
}
func main() {
 u := user{
 name: "stormzhu",
 email: "abc@qq.com",
 }
 sendNotification(u) 
}
// compile error
// cannot use u (type user) as type notifier in argument to sendNotification:
// user does not implement notifier (notify method has pointer receiver)

报的错是u没有实现notifier这个接口,实现了这个接口的是*user类型,而不是user类型,uuser类型,所以不能赋值给notifier这个接口。

既然如此,修改为sendNotification(&u)就OK了。然而问题是,如何理解到底是T类型还是*T类型实现了某个接口呢?

接口的定义

参考雨痕的《Go语言学习笔记》第七章,go语言中的接口定义如下:

type iface struct {
 tab *itab // 类型信息
 data unsafe.Pointer //实际对象指针
}
type itab struct {
 inter *interfacetype // 接口类型
 _type *_type // 实际对象类型
 fun [1]uintptr // 实际对象方法地址
}

虽然具体的细节操作不太懂,但是可以知道,对一个接口赋值的时候,会拷贝类型信息和该类型的方法集。这就类似于C++多态中的虚指针(vptr)和虚函数表(vtable)了。我理解的是,只要这个类型的方法集中包括这个接口的所有方法,那么它就是实现了这个接口,才能够赋值给这个接口,那么问题来了,一个类型的方法集是什么呢?

方法集

同样参考雨痕《Go语言学习笔记》第6章6.3节,书中总结的很全面:

  • 类型T的方法集包含所有 receiver T方法。
  • 类型*T的方法集包含所有 receiver T + *T方法。
  • 匿名嵌入S,类型T的方法集包含所有 receiver T + S方法。
  • 匿名嵌入*S,类型T的方法集包含所有 receiver T + S + *S方法。
  • 匿名嵌入S*S,类型*T的方法集包含所有 receiver T + *T + S + *S方法。

虽然看起来比较复杂,但总结完就一话,*T类型就是厉害,方法集包括T*T的方法。

所以文章开头的例子中,uuser类型,方法集是空的,不算是实现了notifier接口。

当在纠结应该将T类型还是*T类型赋值给某个接口的时候,第一步就是看方法集,看一看该类型到底有没有实现这个接口。(所以T*T不是一个类型。。。)

一些例子

go语言的内置库中有定义了很多接口,如error接口,

type error interface {
 Error() string
}

内置的errors包实现了这个接口:

// Package errors implements functions to manipulate errors.
package errors
// New returns an error that formats as the given text.
func New(text string) error {
 return &errorString{text}
}
// errorString is a trivial implementation of error.
type errorString struct {
 s string
}
func (e *errorString) Error() string {
 return e.s
}

可以看到New方法返回值是error接口,而只有*errorString类型实现了这个接口,所以New方法返回的是&errorString{text}而不是errorString{text}

总结

  • T*T不是一个类型,他们的方法集不同
  • 类型*T的方法集包含所有 receiver T + *T方法,类型T的方法集只包含所有 receiver T方法。

我的SegmentFault博客

参考


有疑问加站长微信联系(非本文作者)

本文来自:简书

感谢作者:StormZhu

查看原文:golang中接口赋值与方法集

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

关注微信
1655 次点击
添加一条新回复 (您需要 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传

用户登录

没有账号?注册
(追記) (追記ここまで)

今日阅读排行

    加载中
(追記) (追記ここまで)

一周阅读排行

    加载中

关注我

  • 扫码关注领全套学习资料 关注微信公众号
  • 加入 QQ 群:
    • 192706294(已满)
    • 731990104(已满)
    • 798786647(已满)
    • 729884609(已满)
    • 977810755(已满)
    • 815126783(已满)
    • 812540095(已满)
    • 1006366459(已满)
    • 692541889

  • 关注微信公众号
  • 加入微信群:liuxiaoyan-s,备注入群
  • 也欢迎加入知识星球 Go粉丝们(免费)

给该专栏投稿 写篇新文章

每篇文章有总共有 5 次投稿机会

收入到我管理的专栏 新建专栏