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 就是错误的。然而,如果重复的名称在类型定义之外的程序中从未被提及,那么这是可以的。这种限定为从外部嵌入的类型的更改提供了一定的保护;如果添加了一个与另一个子类型中的字段冲突的字段,但两个字段都从未被使用,那么就没有问题。

相关推荐
王有品1 小时前
Spring MVC 多个拦截器的执行顺序
数据库·spring·mvc
极小狐2 小时前
如何使用极狐GitLab 的外部状态检查功能?
数据库·ci/cd·gitlab·devops·mcp
言之。2 小时前
Go 语言中的 `select` 语句详解
golang
Leo.yuan2 小时前
数据仓库建设全解析!
大数据·数据库·数据仓库·数据分析·spark
闪电麦坤952 小时前
SQL:子查询(subqueries)
数据库·sql
活跃的煤矿打工人2 小时前
【星海出品】分布式存储数据库etcd
数据库·分布式·etcd
文牧之2 小时前
PostgreSQL的扩展 pgcrypto
运维·数据库·postgresql
_一条咸鱼_3 小时前
揭秘 Android TextInputLayout:从源码深度剖析其使用原理
android·java·面试
_一条咸鱼_3 小时前
揭秘!Android VideoView 使用原理大起底
android·java·面试
_一条咸鱼_3 小时前
深度揭秘!Android TextView 使用原理全解析
android·java·面试