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

相关推荐
Y***h1877 小时前
第二章 Spring中的Bean
java·后端·spring
9***P3347 小时前
PHP代码覆盖率
开发语言·php·代码覆盖率
CoderYanger7 小时前
优选算法-栈:67.基本计算器Ⅱ
java·开发语言·算法·leetcode·职场和发展·1024程序员节
jllllyuz8 小时前
Matlab实现基于Matrix Pencil算法实现声源信号角度和时间估计
开发语言·算法·matlab
稚辉君.MCA_P8_Java8 小时前
DeepSeek 插入排序
linux·后端·算法·架构·排序算法
多多*8 小时前
Java复习 操作系统原理 计算机网络相关 2025年11月23日
java·开发语言·网络·算法·spring·microsoft·maven
t***p9358 小时前
idea创建SpringBoot自动创建Lombok无效果(解决)
spring boot·后端·intellij-idea
d***81728 小时前
解决SpringBoot项目启动错误:找不到或无法加载主类
java·spring boot·后端
p***43488 小时前
Rust网络编程模型
开发语言·网络·rust
无限大68 小时前
RBAC模型:像电影院选座一样管理权限,告别"一个用户配一个权限"的噩梦
后端