go结构体匿名“继承“方法冲突时继承优先顺序

在 Go 语言中,匿名字段(也称为嵌入字段)可以用来实现继承的效果。当你在一个结构体中匿名嵌入另一个结构体时,嵌入结构体的方法会被提升到外部结构体中。这意味着你可以直接通过外部结构体调用嵌入结构体的方法。

如果多个嵌入结构体实现了同一个接口方法,那么调用时会根据方法的定义顺序来决定调用哪个方法。具体来说,Go 语言会选择第一个定义的方法。

示例

假设我们有两个结构体 AB,它们都实现了同一个接口 MyInterface,然后我们在结构体 C 中匿名嵌入了这两个结构体。

go 复制代码
package main

import "fmt"

// MyInterface 接口
type MyInterface interface {
	DoSomething()
}

// 结构体 A
type A struct{}

func (a A) DoSomething() {
	fmt.Println("A.DoSomething()")
}

// 结构体 B
type B struct{}

func (b B) DoSomething() {
	fmt.Println("B.DoSomething()")
}

// 结构体 C 匿名嵌入了 A 和 B
type C struct {
	A
	B
}

func main() {
	c := C{}
	var myInterface MyInterface = c

	myInterface.DoSomething() // 输出什么?
}

输出结果

在这个例子中,调用 myInterface.DoSomething() 会输出:

复制代码
A.DoSomething()

解释

  • 方法提升 :当 C 结构体匿名嵌入了 AB 时,AB 的方法都被提升到了 C 中。
  • 方法冲突 :由于 AB 都实现了 DoSomething 方法,因此 C 中会有两个同名的方法。
  • 方法选择 :在 Go 语言中,当多个嵌入字段中有同名方法时,会优先选择第一个定义的方法。在这个例子中,A 是第一个被嵌入的字段,因此 ADoSomething 方法会被调用。

更多示例

为了进一步说明这一点,我们可以添加更多的嵌入字段来观察方法的选择顺序。

go 复制代码
package main

import "fmt"

// MyInterface 接口
type MyInterface interface {
	DoSomething()
}

// 结构体 A
type A struct{}

func (a A) DoSomething() {
	fmt.Println("A.DoSomething()")
}

// 结构体 B
type B struct{}

func (b B) DoSomething() {
	fmt.Println("B.DoSomething()")
}

// 结构体 D
type D struct{}

func (d D) DoSomething() {
	fmt.Println("D.DoSomething()")
}

// 结构体 C 匿名嵌入了 A、B 和 D
type C struct {
	A
	B
	D
}

func main() {
	c := C{}
	var myInterface MyInterface = c

	myInterface.DoSomething() // 输出什么?
}

输出结果

在这个例子中,调用 myInterface.DoSomething() 会输出:

复制代码
A.DoSomething()

解释

  • 方法提升ABDDoSomething 方法都被提升到了 C 中。
  • 方法选择 :由于 A 是第一个被嵌入的字段,因此 ADoSomething 方法会被优先调用。

扩展

当继承类本身也实现了对应方法时,优先使用本身实现的方法

go 复制代码
// MyInterface 接口
type MyInterface interface {
	DoSomething() string
}

// 结构体 A
type A struct{}

func (a A) DoSomething() string {
	fmt.Println("A.DoSomething()")
	return "A"
}

// 结构体 B
type B struct{}

func (b B) DoSomething() string {
	fmt.Println("B.DoSomething()")
	return "B"
}

// 结构体 C 匿名嵌入了 A 和 B
type C struct {
	A
	B
}

func (c C) DoSomething() string {
	fmt.Println("C.DoSomething()")
	return "C"
}

func TestEnhanceStrcutC(t *testing.T) {
	var baseC C
	// 这里将会调用func (c C) DoSomething() string
	if "C" != baseC.DoSomething() {
		t.Error("DoSomething failed.")
	}
}

总结

在 Go 语言中,当一个结构体匿名嵌入了多个实现相同接口的结构体时,调用该接口方法时会优先选择自己实现的方法,如果自己没有实现该方法,就按照顺序从上到下找到第一个定义的方法。方法的定义顺序决定了调用哪个方法。但是为了避免歧义和提高代码的可读性,建议在设计时尽量避免这种情况,或者在外部结构体中显式地实现接口方法。

相关推荐
方璧18 小时前
限流的算法
java·开发语言
Hi_kenyon19 小时前
VUE3套用组件库快速开发(以Element Plus为例)二
开发语言·前端·javascript·vue.js
曲莫终19 小时前
Java VarHandle全面详解:从入门到精通
java·开发语言
奋进的芋圆19 小时前
DataSyncManager 详解与 Spring Boot 迁移指南
java·spring boot·后端
ghie909019 小时前
基于MATLAB GUI的伏安法测电阻实现方案
开发语言·matlab·电阻
Gao_xu_sheng19 小时前
Inno Setup(专业安装/更新 EXE)
开发语言
计算机程序设计小李同学19 小时前
个人数据管理系统
java·vue.js·spring boot·后端·web安全
Echo娴20 小时前
Spring的开发步骤
java·后端·spring
追逐时光者20 小时前
TIOBE 公布 C# 是 2025 年度编程语言
后端·.net