这些技巧你必须知道,Go语言模拟继承顶级指南

简介: 这些技巧你必须知道,Go语言模拟继承顶级指南

/ Go 语言结构体内嵌模拟类的继承 /

 

一、概述

Go 语言中不存在传统面向对象中的类继承,但可以通过结构体内嵌来模拟类的继承。本文将介绍如何使用结构体内嵌来设计类继承关系,实现代码复用。

主要内容包括:

  • 类继承基本概念
  • Go 语言结构体内嵌语法
  • 内嵌结构体与组合结构体
  • 初始化内嵌结构体
  • 方法覆盖与扩展
  • 多重继承实现
  • 内嵌与继承的差异
  • 实际应用案例
  • 封装与继承关系

通过学习这些知识,将能够自如地使用 Go 语言的内嵌机制来设计类继承,编写出更实用的 Go 代码。


 

二、类继承基本概念

类继承是面向对象编程中的一个核心概念,它允许子类继承父类的数据字段和方法,从而实现复用。

子类是一个更具体的类,父类是一个更通用的类。继承可以使子类获得父类的全部或部分功能。

Go 语言通过结构体内嵌可以实现类似的继承关系。


 

三、Go 语言结构体内嵌语法

Go 语言使用结构体内嵌来模拟类的继承,其语法如下:

type Parent struct {
  // 父类字段和方法
} 
type Child struct {
  Parent // 继承Parent
  // 子类字段和方法
}

Child 结构体通过嵌入 Parent 结构体,就可以获得 Parent 的字段和方法。这实现了继承关系。


 

四、内嵌结构体与组合结构体

结构体内嵌看起来类似组合结构体:

type Child struct {
  parent Parent
}

但其实内嵌与组合有以下差异:

  • 内嵌可以直接访问 Parent 字段和方法,组合需要通过实例名访问
  • 内嵌创建的是 is-a 关系,组合是 has-a 关系

一般来说,内嵌更能表达继承关系。


 

五、初始化内嵌结构体

初始化内嵌结构体需要指定内嵌成员:

type Parent struct {
  name string
}
type Child struct {
  Parent
}
func main() {
  c := Child{
    Parent{name: "Tom"}, 
  }
}

如果省略 Parent,则会使用默认值初始化。


 

六、方法覆盖与扩展

子类可以定义与父类同名的方法,这相当于方法 Override:

type Parent struct {}
func (p *Parent) test() {}
type Child struct {
  Parent
}
// 覆盖Parent的test方法
func (c *Child) test() {}


这里 Child 实现了自己的 test 方法,覆盖了 Parent 的 test 方法。

如果要扩展父类方法,可以在方法中调用父类的实现:

func (c *Child) test() {
  // 先调用Parent的实现
  c.Parent.test() 
  // Child的处理逻辑
}

七、多重继承实现

Go 语言可以通过多重内嵌实现多重继承:

type A struct {}
type B struct {}
type C struct {
  A
  B
}


结构体 C 通过嵌入 A 和 B,实现了 A、B 两者的多重继承。


 

八、内嵌与继承的差异

尽管内嵌模拟了继承,但与传统面向对象继承还有一些区别:

  • Go 没有类型继承,只能结构体内嵌
  • 内嵌不会继承私有成员
  • 内嵌采取就近访问,非线性查找

内嵌与传统面向对象继承有些细微差异,来看一个例子:

package main
import "fmt"
type Animal struct {
  name string
}
func (a *Animal) Eat() {
  fmt.Printf("%s is eating\n", a.name)
}
type Dog struct {
  Animal
}
func main() {
  d := Dog{Animal{name: "Wangcai"}}
  d.Eat()
  // 内嵌不继承私有字段和方法
  // d.privateCannotAccess ()  
}


在设计时需要注意这些差异。


 

九、实际应用案例

我们可以基于内嵌实现一些常见的设计模式:

  • 多态:接口+内嵌实现
  • 装饰器:内嵌扩展功能
  • 代理:内嵌实现代理类
  • 桥接:内嵌将抽象与实现解耦


 

1、装饰器模式:

package main
import "fmt"
type Shape interface {
  Draw()
}
type Circle struct {
  radius float32
}
func (c *Circle) Draw() {
  fmt.Printf("Draw circle, radius=%f\n", c.radius)
}
type ColoredCircle struct {
  Circle 
  color string
}
func (c *ColoredCircle) Draw() {
  fmt.Printf("Draw colored circle, color=%s\n", c.color)
  c.Circle.Draw()
}
func main() {
  cc := &ColoredCircle{
    Circle{10}, "Red"}
  cc.Draw()
}

2、多态,使用接口+内嵌可以实现多态

type Sayer interface {
    Say()
}
type Dog struct {}
func (d *Dog) Say() {
    println("Wang!") 
}
type Cat struct {}
func (c *Cat) Say() {
    println("Miao!")
}
func main() {
    animals := []Sayer{
        &Dog{},
        &Cat{},
    }
    for _, animal := range animals {
        animal.Say() 
    }
}

3、代理

package main
import "fmt"
type Image interface {
    Display()
}
type RealImage struct{
    filename string
}
func (i *RealImage) Display() {
    fmt.Println("Displaying", i.filename)
}
type ProxyImage struct {
    RealImage
}
func (p *ProxyImage) Display() {
    if p.RealImage.filename == "" {
        p.RealImage.filename = "test.jpg" // 初始化filename
    }
    fmt.Println("Proxy displaying", p.RealImage.filename) 
    p.RealImage.Display()
}
func main() {
    image := &ProxyImage{}
    image.Display()
}

4、桥接,桥接可动态改变渲染器实现

package main
import "fmt"
type Renderer interface {
    RenderCircle(radius float32)
}
type VectorRenderer struct{}
func (v *VectorRenderer) RenderCircle(radius float32) {
    fmt.Println("Drawing a circle of radius", radius)
}
type RasterRenderer struct{}
func (r *RasterRenderer) RenderCircle(radius float32) {
    fmt.Println("Drawing pixels for circle of radius", radius) 
}
type Circle struct {
    renderer Renderer
    radius float32
}
func (c *Circle) Draw() {
    c.renderer.RenderCircle(c.radius)
}
func main() {
    raster := &RasterRenderer{}
    vector := &VectorRenderer{}
    circle := &Circle{vector, 5}
    circle.Draw()
    circle.renderer = raster
    circle.Draw()
}

这些设计模式可以使程序具有更好的扩展性。


 

十、封装与继承关系

继承可以通过访问控制来保持良好的封装关系:

type Parent struct {
  a int // 私有字段
  A int // 公开字段
}
type Child struct {
  Parent
  b int
}
func main() {
  c := Child{}
  c.A = 1 // 可以访问父类公开字段
  c.a = 2 // 不能访问父类私有字段
}

通过这种方式,继承可以在保持封装的前提下实现复用。


 

十一、总结

Go 语言通过结构体内嵌支持了类似面向对象继承的特性,这有助于复用和扩展代码。但与传统继承还存在一些差异需要注意。

合理利用结构体内嵌機制可以编写出更清晰实用的 Go 代码。这是 Go 语言实现代码复用的核心方式之一。


目录
相关文章
|
3天前
|
存储 监控 算法
员工上网行为监控中的Go语言算法:布隆过滤器的应用
在信息化高速发展的时代,企业上网行为监管至关重要。布隆过滤器作为一种高效、节省空间的概率性数据结构,适用于大规模URL查询与匹配,是实现精准上网行为管理的理想选择。本文探讨了布隆过滤器的原理及其优缺点,并展示了如何使用Go语言实现该算法,以提升企业网络管理效率和安全性。尽管存在误报等局限性,但合理配置下,布隆过滤器为企业提供了经济有效的解决方案。
30 8
员工上网行为监控中的Go语言算法:布隆过滤器的应用
|
23天前
|
存储 Go 索引
go语言中数组和切片
go语言中数组和切片
36 7
|
22天前
|
Go 开发工具
百炼-千问模型通过openai接口构建assistant 等 go语言
由于阿里百炼平台通义千问大模型没有完善的go语言兼容openapi示例,并且官方答复assistant是不兼容openapi sdk的。 实际使用中发现是能够支持的,所以自己写了一个demo test示例,给大家做一个参考。
|
23天前
|
程序员 Go
go语言中结构体(Struct)
go语言中结构体(Struct)
97 71
|
22天前
|
存储 Go 索引
go语言中的数组(Array)
go语言中的数组(Array)
102 67
|
23天前
|
存储 Go
go语言中映射
go语言中映射
35 11
|
24天前
|
Go 索引
go语言使用索引遍历
go语言使用索引遍历
29 9
|
24天前
|
Go 索引
go语言使用range关键字
go语言使用range关键字
29 7
|
24天前
|
Go 索引
go语言修改元素
go语言修改元素
29 6
|
14天前
|
Go 数据安全/隐私保护 UED
优化Go语言中的网络连接:设置代理超时参数
优化Go语言中的网络连接:设置代理超时参数