Go语言使用组合的思想实现继承

前言

  • Go 语言的设计之初,就不打算支持面向对象的编程特性,因此 Go 不支持面向对象的三大特性之一——继承。但是 Go 可以通过组合的思想去实现 “继承”。
  • 继承是面向对象的三大特性之一,继承是从已有的类中派生出新的类,新的类能吸收已有类的数据属性和行为,并能扩展新的能力。
  • Go 语言里的“继承”体现如一个结构体拥有另一个结构体的的所有字段和方法,并在此基础上,定义新的字段和方法。

类型嵌入

组合的思想,在 Go 语言里的体现就是类型嵌入。类型嵌入,指的是一个类型的定义里嵌入了其他类型。类型嵌入包含两种,一种是结构体类型嵌入,另一种是接口类型嵌入。

结构体类型嵌入

import "fmt"

type Person struct {
    Name string
    Age  int
}

func (p Person) Introduce() {
    fmt.Printf("大家好,我叫%s,我今年%d岁了。\n", p.Name, p.Age)
}

type Student struct {
    Person
    School string
}

func (s Student) GoToTheClass() {
    fmt.Println("去上课...")
}

func main() {
    student := Student{}
    student.Name = "小明"
    student.Age = 18
    student.School = "太阳系大学"

    // 执行 Person 类型的 Introduce 方法
    student.Introduce()
    // 执行自身的 GoToTheClass 方法
    student.GoToTheClass()
}

执行结果:

大家好,我叫小明,我今年18岁了。

去上课...

  • 定义 Person 结构体,包含 NameAge 字段以及 Introduce 方法。
  • 定义 Student 结构体,定义 School 字段,GoToTheClass 方法,嵌入 Person 类型,直接以结构体类型作为结构体的字段,称为 结构体的类型嵌入,此字段也被视为 字段嵌入
  • 创建 Student 变量 student,通过 student 访问 NameAge 字段以及 Introduce 方法,明明 Student 没有定义这两个字段和这个方法,为什么能访问和使用呢?这是因为 Student “继承” 了 PersonPerson 结构体的字段和方法,在 Student 结构体里隐式实现。

接口类型嵌入

type Coder interface {
    Code()
}

type Tester interface {
    Test()
}

type TesterCoder interface {
    Tester
    Coder
}
  • 定义 Coder 接口,接口有 Code 编码方法。
  • 定义 Tester 接口,接口有 Test 测试方法。
  • 定义 TesterCoder 接口,内嵌 CoderTester 接口类型,通过组合,让 TesterCoder 接口拥有 Code 编码方法和 Test 测试方法。看到这里,有的小伙伴可能会想到开发岗测试岗测试开发岗
  • Go 标准库中可以看到很多这种组合方式的应用。

小结

本文介绍了 Go 语言中的 “继承”,它是通过组合的思想去模拟实现面向对象中的继承。然后介绍了什么是类型嵌入以及类型嵌入的两种类型,嵌入的类型包含的字段和方法以隐式存在。

“继承”的实现,能够提高代码的复用性,代码的维护性和扩展性也得以提高。

原文地址:https://juejin.cn/post/7177405287446839354