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

相关推荐
青春_strive3 分钟前
Qt常用控件之显示类控件
开发语言·qt
2401_857636396 分钟前
宠物领养平台建设:SpringBoot案例分析
spring boot·后端·宠物
布朗克1688 分钟前
JWT介绍和结合springboot项目实践(登录、注销授权认证管理)
java·spring boot·后端·安全·jwt
城沐小巷1 小时前
外卖点餐系统小程序
前端·后端·微信小程序
途途途途1 小时前
100个python经典面试题详解(新版)
开发语言·python·最新面试题·python面试题
uhakadotcom1 小时前
开发我们的第一个基于Rust的应用:端口扫描器
后端·安全·rust
以卿a1 小时前
C++ 类和对象(类型转换、static成员)
开发语言·c++·算法
Muisti2 小时前
P7184 [CRCI2008-2009] MAJSTOR 多层循环的遍历
开发语言·c++·算法·leetcode
晚渔声2 小时前
【线程】Java多线程代码案例(2)
java·开发语言·多线程
5-StarrySky2 小时前
Java 线程中的分时模型和抢占模型
java·开发语言