当前位置: 首页 > news >正文

golang中接口赋值与方法集

接口使用疑问

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方法。

我的简书博客

参考

  • Go语言学习笔记
  • Go语言实战

相关文章:

  • tensorflow 戴明线性回归
  • 如果我告诉你,程序员这条路很难走,你还要坚持走下去吗
  • A Tour of Go: Basics 2
  • Scala学习(十)特质
  • linux安装LNMP环境之安装PHP
  • Graphviz的安装 - windows环境下
  • RSocket:一个面向反应式应用程序的新型应用网络协议
  • Spring Bean生命周期-registerBeanPostProcessors(七)
  • C#指定用户启动进程
  • ServletRequest和ServletResponse学习笔记
  • Ubuntu16.04 System program problem detected
  • MariaDB10.3 增补AliSQL补丁---安全执行Online DDL
  • shell脚本中打印所有匹配某些关键字符的行或前后各N行
  • 数组遍历的方法(loop)
  • 18-10-11
  • 自己简单写的 事件订阅机制
  • 【JavaScript】通过闭包创建具有私有属性的实例对象
  • 【跃迁之路】【519天】程序员高效学习方法论探索系列(实验阶段276-2018.07.09)...
  • 2017-08-04 前端日报
  • CentOS7简单部署NFS
  • JS专题之继承
  • React的组件模式
  • SpingCloudBus整合RabbitMQ
  • SpringCloud(第 039 篇)链接Mysql数据库,通过JpaRepository编写数据库访问
  • uni-app项目数字滚动
  • 初识 webpack
  • 目录与文件属性:编写ls
  • 前端知识点整理(待续)
  • 新书推荐|Windows黑客编程技术详解
  • 一起参Ember.js讨论、问答社区。
  • 白色的风信子
  • 昨天1024程序员节,我故意写了个死循环~
  • ​MPV,汽车产品里一个特殊品类的进化过程
  • # centos7下FFmpeg环境部署记录
  • #git 撤消对文件的更改
  • (3)STL算法之搜索
  • (C#)Windows Shell 外壳编程系列4 - 上下文菜单(iContextMenu)(二)嵌入菜单和执行命令...
  • (delphi11最新学习资料) Object Pascal 学习笔记---第2章第五节(日期和时间)
  • (NO.00004)iOS实现打砖块游戏(九):游戏中小球与反弹棒的碰撞
  • (Repost) Getting Genode with TrustZone on the i.MX
  • (二)WCF的Binding模型
  • (分类)KNN算法- 参数调优
  • (四)汇编语言——简单程序
  • (译) 理解 Elixir 中的宏 Macro, 第四部分:深入化
  • (原创) cocos2dx使用Curl连接网络(客户端)
  • (转) 深度模型优化性能 调参
  • (转)创业的注意事项
  • ***测试-HTTP方法
  • .bat批处理(十):从路径字符串中截取盘符、文件名、后缀名等信息
  • .cfg\.dat\.mak(持续补充)
  • .gitignore文件—git忽略文件
  • .mat 文件的加载与创建 矩阵变图像? ∈ Matlab 使用笔记
  • .net 7 上传文件踩坑
  • .NET Core实战项目之CMS 第一章 入门篇-开篇及总体规划
  • .NET Framework Client Profile - a Subset of the .NET Framework Redistribution