第 71 期周刊题目 — 结构体内嵌接口:65% 的人答错了

Go语言精选

共 2183字,需浏览 5分钟

 ·

2020-12-22 13:59

大家好,我是站长 polarisxu。

时不时有人催问周刊题目的解析,那就先先解析下第 71 期的题目。

01

题目是,以下代码能否通过编译:

package main

import (
 "fmt"
)

type worker interface {
 work()
}

type person struct {
 name string
 worker
}

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

这是大家的投票结果:

只有 35% 的人答对了。

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

02

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

type person struct {
 name string
 worker worker
}

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

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

var w worker = person{}

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

03

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

在 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 intbool
 // Swap swaps the elements with indexes i and j.
 Swap(i, j int)
}

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

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)
}

04

回到上面的题目,如果我们通过实例化的 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"}}

你掌握了吗?


对了,昨天发的 Gopher Dinner,不少人对其中的 Gopher 表情包感兴趣。和永博沟通后,一起整理完了,现在分享给大家。

关注下方公众号,回复「表情」获取。

参考资料

[1]

reverse: https://docs.studygolang.com/src/sort/sort.go?s=7078:7105#L239



往期推荐



福利

我为大家整理了一份从入门到进阶的Go学习资料礼包,包含学习建议:入门看什么,进阶看什么。关注公众号 「polarisxu」,回复 ebook 获取;还可以回复「进群」,和数万 Gopher 交流学习。

浏览 21
点赞
评论
收藏
分享

手机扫一扫分享

举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

举报