
本文探讨了在 Go 语言中,当一个类型通过类型转换被当作另一个类型使用时,如何在被调用的方法内部判断其原始类型。核心结论是,通过类型转换后,原始类型的信息会丢失。文章提供了一种使用接口的替代方案,允许在方法内部区分不同的实现类型。
在 Go 语言中,类型系统与面向对象语言有所不同。当我们基于一个类型创建新的类型时,并不会像面向对象语言那样建立继承关系。这意味着,如果将一个类型 T1 转换为类型 T,那么在 T 的方法中,无法直接得知它是由 T1 转换而来。
问题描述
假设我们有类型 T,以及基于 T 创建的类型 T1 和 T2。我们希望在 T 的方法 WhoAmI() 中判断调用该方法的实例究竟是 T1 还是 T2 转换而来。
立即学习“go语言免费学习笔记(深入)”;
package main
import "fmt"
import "reflect"
type T struct{ s string }
func (v *T) WhoAmI() string {
// pull type name with reflect
fmt.Println(reflect.TypeOf(v).Elem().Name()) // always prints "T"!
// todo: if I am actually T1
return "T1"
// todo: else if I am actually T2
return "T2"
}
type T1 T
func NewT1(s string) T1 { return T1{s} }
type T2 T
func NewT2(s string) T2 { return T2{s} }
func main() {
var t1 = T1{"xyz"}
var t2 = T2{"pdq"}
s1 := ((*T)(&t1)).WhoAmI() // would like to return "T1"
s2 := ((*T)(&t2)).WhoAmI() // would like to return "T2"
fmt.Println(s1, s2)
}上述代码中,WhoAmI() 方法总是返回 "T",因为在类型转换后,原始类型的信息已经丢失。
解决方案:使用接口
Go 语言提供了一种更灵活的方式来实现类似的功能,即使用接口。我们可以定义一个接口 T,然后让 T1 和 T2 都实现这个接口。
type T interface {
WhoAmI() string
}
type T1 struct {
s string
}
func (t *T1) WhoAmI() string { return "T1" }
type T2 struct {
s string
}
func (t *T2) WhoAmI() string { return "T2" }
func main() {
var t1 T = &T1{"xyz"}
var t2 T = &T2{"pdq"}
fmt.Println(t1.WhoAmI(), t2.WhoAmI())
}在这个例子中,T1 和 T2 都实现了接口 T。因此,我们可以将 T1 和 T2 的实例赋值给类型为 T 的变量。当我们调用 WhoAmI() 方法时,会根据实际的类型执行对应的方法,从而得到正确的结果。
示例代码解释
注意事项
总结
在 Go 语言中,如果需要根据类型转换前的原始类型来执行不同的逻辑,应该使用接口来实现。通过接口,我们可以实现多态,并在运行时根据实际类型选择对应的方法执行。这是一种更灵活、更强大的方式来处理类型相关的逻辑。
以上就是Go语言中如何判断通过类型转换调用的方法的原始类型?的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号