【发布时间】:2019-01-31 22:29:17
【问题描述】:
我一直在使用 Golang 中的接口和结构来尝试实现“继承”,但我确信我做错了。举个例子会更容易解释。
我想创造不同生物的结构。我希望他们拥有 GetName() 方法:
type LivingThingProvider interface {
GetName() string
}
现在,他们都应该有名字和生日。为此,我正在创建一个结构并将接口嵌入其中:
type LivingThing struct {
birthday string
name string
LivingThingProvider
}
我想添加几个对所有生物都一样的方法:
func (this *LivingThing) Initialize() {
this.birthday = time.Now().Format("02.01.2006")
}
func (this LivingThing) GetBirthday() string {
return this.birthday
}
现在,这里是应该“实现” LivingThing 的结构:
type Frog struct {
insectsEaten int
LivingThing
}
func (this Frog) GetName() string {
return fmt.Sprintf("%s-the-Frog-proud-eater-of-%d-insects", this.name, this.insectsEaten)
}
type RobotMan struct {
LivingThing
}
func (this RobotMan) GetName() string {
h := sha256.New()
h.Write([]byte(this.birthday))
return fmt.Sprintf("%s-%X", this.name, h.Sum(nil))
}
在主函数中,我正在创建一个青蛙和一个机器人并将其添加到切片中,然后循环遍历它:
func main() {
fr := Frog{}
fr.name = "Dizzy"
fr.insectsEaten = 586
fr.LivingThingProvider = fr
rm := RobotMan{}
rm.name = "Bender"
rm.LivingThingProvider = rm
fr.Initialize()
rm.Initialize()
entities := []LivingThing{fr.LivingThing, rm.LivingThing}
for _, ent := range entities {
fmt.Printf("Hi, I am %s!\n", ent.GetName())
fmt.Printf("I was born on the %s.\n", ent.GetBirthday())
}
}
一切都按预期工作,但如果我从 Frog 或 RobotMan 结构中删除 GetName() 方法,它将在运行后编译并崩溃:
panic: runtime error: invalid memory address or nil pointer dereference
这里是操场链接:https://play.golang.org/p/h2VgvdcXJQA
我的问题如下:
1. 就 Go 而言,我所做的是否“肮脏”?如果是这样,如何以正确的方式做到这一点?
1a. 特别是,将结构本身分配给其嵌入式接口字段(fr.LivingThingProvider = fr)是否可以?
2.为什么 Go 编译器不检查 Frog 和 RobotMan 是否实现 LivingThingProvider 接口?
非常感谢您!
【问题讨论】:
-
1. 是的,尝试“实现”继承是编写 Go 代码的错误方式。正确的做法是使用组合。
-
2. 这两种类型仍然实现了接口,因为它们嵌入了 LivingThing,它嵌入了未设置的 LivingThingProvider,因此为 nil,因此您会感到恐慌。如果结构类型嵌入了另一种类型,则嵌入类型的方法可以通过嵌入类型直接访问,这就是为什么即使删除了它们的方法,这两种类型仍然实现接口的原因。
-
... 还要注意,接口是隐式满足的。您无需嵌入接口即可让嵌入类型实现它。您可以通过在所需实现者上声明与接口上声明的签名相同的方法来实现接口。
-
@mkopriva,非常感谢!经过您的解释,它似乎变得如此简单明了。再次感谢您。
标签: inheritance go interface embedding