go-基础之嵌入

嵌入

Go 语言没有提供典型的、基于类型驱动的子类化概念,但它能够通过在结构体或接口中嵌入类型来"借用"部分实现。

接口嵌入非常简单。我们之前提到过 io.Readerio.Writer 接口,下面是它们的定义:

go 复制代码
type Reader interface {
    Read(p []byte) (n int, err error)
}

type Writer interface {
    Write(p []byte) (n int, err error)
}

io 包还导出了其他几个接口,这些接口定义的对象可以实现多个这样的方法。例如,有 io.ReadWriter 接口,它包含 ReadWrite 方法。我们可以通过显式列出这两个方法来定义 io.ReadWriter,但更简单且更具表现力的方式是嵌入这两个接口来形成新的接口,如下所示:

go 复制代码
// ReadWriter 是一个组合了 Reader 和 Writer 接口的接口。
type ReadWriter interface {
    Reader
    Writer
}

这正如它看起来的那样:ReadWriter 既可以执行 Reader 的操作,也可以执行 Writer 的操作;它是所嵌入接口的集合。只有接口才能嵌入到接口中。

同样的基本概念也适用于结构体,但会有更深远的影响。bufio 包有两个结构体类型,bufio.Readerbufio.Writer,当然,它们各自实现了 io 包中对应的接口。bufio 包还实现了一个带缓冲的读写器,它通过嵌入将一个读取器和一个写入器组合到一个结构体中来实现这一点:在结构体中列出类型但不指定字段名。

go 复制代码
// ReadWriter 存储了指向 Reader 和 Writer 的指针。
// 它实现了 io.ReadWriter 接口。
type ReadWriter struct {
    *Reader  // *bufio.Reader
    *Writer  // *bufio.Writer
}

嵌入的元素是指向结构体的指针,当然,在使用它们之前必须将其初始化为指向有效的结构体。ReadWriter 结构体也可以写成这样:

go 复制代码
type ReadWriter struct {
    reader *Reader
    writer *Writer
}

但这样的话,为了提升字段的方法并满足 io 接口,我们还需要提供转发方法,如下所示:

go 复制代码
func (rw *ReadWriter) Read(p []byte) (n int, err error) {
    return rw.reader.Read(p)
}

通过直接嵌入结构体,我们避免了这种额外的处理。嵌入类型的方法会自动成为外部类型的方法,这意味着 bufio.ReadWriter 不仅拥有 bufio.Readerbufio.Writer 的方法,还满足所有三个接口:io.Readerio.Writerio.ReadWriter

嵌入与子类化有一个重要的区别。当我们嵌入一个类型时,该类型的方法会成为外部类型的方法,但在调用这些方法时,方法的接收者是内部类型,而不是外部类型。在我们的示例中,当调用 bufio.ReadWriterRead 方法时,其效果与上面写出的转发方法完全相同;接收者是 ReadWriterreader 字段,而不是 ReadWriter 本身。

嵌入也可以是一种简单的便利方式。下面这个示例展示了一个嵌入字段和一个常规的命名字段。

go 复制代码
type Job struct {
    Command string
    *log.Logger
}

Job 类型现在拥有 *log.LoggerPrintPrintfPrintln 等方法。当然,我们也可以给 Logger 一个字段名,但这不是必需的。现在,一旦初始化完成,我们就可以对 Job 进行日志记录:

go 复制代码
job.Println("starting now...")

LoggerJob 结构体的一个常规字段,所以我们可以在 Job 的构造函数中以通常的方式对其进行初始化,如下所示:

go 复制代码
func NewJob(command string, logger *log.Logger) *Job {
    return &Job{command, logger}
}

或者使用复合字面量:

go 复制代码
job := &Job{command, log.New(os.Stderr, "Job: ", log.Ldate)}

如果我们需要直接引用嵌入字段,忽略包限定符的字段类型名可以作为字段名,就像在 ReadWriter 结构体的 Read 方法中那样。在这里,如果我们需要访问 Job 变量 job*log.Logger,我们可以写成 job.Logger,如果我们想改进 Logger 的方法,这会很有用。

go 复制代码
func (job *Job) Printf(format string, args ...interface{}) {
    job.Logger.Printf("%q: %s", job.Command, fmt.Sprintf(format, args...))
}

嵌入类型会引入名称冲突的问题,但解决这些问题的规则很简单。首先,字段或方法 X 会隐藏类型中更深嵌套部分的任何其他 X 项。如果 log.Logger 包含一个名为 Command 的字段或方法,JobCommand 字段会覆盖它。

其次,如果相同的名称出现在同一嵌套级别,通常这是一个错误;如果 Job 结构体包含另一个名为 Logger 的字段或方法,嵌入 log.Logger 就是错误的。然而,如果重复的名称在类型定义之外的程序中从未被提及,那么这是可以的。这种限定为从外部嵌入的类型的更改提供了一定的保护;如果添加了一个与另一个子类型中的字段冲突的字段,但两个字段都从未被使用,那么就没有问题。

相关推荐
嘴对嘴编程28 分钟前
oracle数据泵操作
数据库·oracle
冷琅辞1 小时前
Go语言的嵌入式网络
开发语言·后端·golang
response_L1 小时前
国产系统统信uos和麒麟v10在线打开word给表格赋值
java·c#·word·信创·在线编辑
苹果酱05671 小时前
Golang标准库——runtime
java·vue.js·spring boot·mysql·课程设计
User_芊芊君子1 小时前
【Java】类和对象
java·开发语言
martian6651 小时前
Spring Boot后端开发全攻略:核心概念与实战指南
java·开发语言·spring boot
跟着珅聪学java3 小时前
spring boot +Elment UI 上传文件教程
java·spring boot·后端·ui·elementui·vue
我命由我123453 小时前
Spring Boot 自定义日志打印(日志级别、logback-spring.xml 文件、自定义日志打印解读)
java·开发语言·jvm·spring boot·spring·java-ee·logback
lilye663 小时前
程序化广告行业(55/89):DMP与DSP对接及数据统计原理剖析
java·服务器·前端
徐小黑ACG4 小时前
GO语言 使用protobuf
开发语言·后端·golang·protobuf