Golang知识点记录 -- 接口嵌套

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

以下代码能否通过编译?

package main

import (
	"fmt"
)

type worker interface {
	work()
}

type person struct {
	name string
	worker
}

func main() {
	var w worker = person{}
	fmt.Println(w)
}

A: 能;B: 不能;C: 不知道

 

答案解析:

这里答错的原因在于 worker 是一个接口,如果是一个普通的类型,相信大家会答对。一个结构体竟然可以嵌入一个接口?!

01

我们都知晓 Go 没有继承,但可以通过内嵌类型模拟部分继承的功能。大家要记住,接口也是类型,自然也将它作为嵌入类型。如果题目的 person 接口体改为:

type person struct {
	name string
	worker worker
}

相信会有更多人答对,这和嵌入类型唯一的区别在于是否显示指定了字段名,其他并无区别。

将接口作为嵌入类型可能让人感觉有些奇怪:那这个类型不是默认就实现了这个接口?!确实是这样的,所以才有了题目中这一句能编译通过:

var w worker = person{}

只不过,因为实例化 person 时,没有给 worker 指定值,因此 person 中的 worker 是 nil,调用它的话会报错,但编译是没问题的。

02

有人可能要问,嵌入接口有实际用途吗?我找一个标准库中的例子。

在 sort 包中,有一个接口:Interface

type Interface interface {
	// Len is the number of elements in the collection.
	Len() int
	// Less reports whether the element with
	// index i should sort before the element with index j.
	Less(i, j int) bool
	// Swap swaps the elements with indexes i and j.
	Swap(i, j int)
}

这是用于排序的。还有另外一个结构体:reverse

type reverse struct {
	// This embedded Interface permits Reverse to use the methods of
	// another Interface implementation.
	Interface
}

它就内嵌了一个 Interface,用于排序的反转。

而内嵌接口的关键在于如何给这个内嵌的接口赋值。sort 包有一个函数:Reverse

func Reverse(data Interface) Interface {
	return &reverse{data}
}

其中实例化 reverse 时,直接通过传递的 Interface 实例赋值给 reverse 的内嵌接口,然后 reverse 类型可以有选择的重新实现内嵌的 Interface 的方法。比如 Less 方法:

func (r reverse) Less(i, j int) bool {
	return r.Interface.Less(j, i)
}

03

回到上面的题目,如果我们通过实例化的 w 调用 work 方法会报错:

var w worker = person{}
w.work()	// panic

和上面 reverse 类似,你需要给 person 中的 worker 实例化,也就是需要一个实现了 worker 接口的类型实例。比如:

type student struct{
    name string
}

func (s student) work() {
    fmt.Println("I am ", s.name, ", I am learning")
}

然后这样实例化 person:

var w worker = person{worker: student{"polarisxu"}}

 

以上内容来自:Go每日一题(2022-02-23) 的题目

 

 


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

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

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