登录
首页 >  Golang >  Go问答

Golang中如何使用reflect和interface{}实现带参方法的多态?

来源:stackoverflow

时间:2024-02-27 20:33:24 405浏览 收藏

哈喽!今天心血来潮给大家带来了《Golang中如何使用reflect和interface{}实现带参方法的多态?》,想必大家应该对Golang都不陌生吧,那么阅读本文就都不会很困难,以下内容主要涉及到,若是你正在学习Golang,千万别错过这篇文章~希望能帮助到你!

问题内容

我正在将一些代码从 c++ 移植到 golang。我对golang中的oop和多态性做了一些研究,都建议使用接口和嵌入来实现多态性。我有一个示例代码编写如下

type worker interface {
    show()
    inc()
}

type workera struct {
    x int
    worker // embed all methods in interface
}

type workerb struct {
    workera // embed workera to have all methods
}

func (a *workera) inc() {
    a.x++
}

func (a workera) show() {
    fmt.println("a:", a.x)
}

func (b workerb) show() {
    fmt.println("b:", b.x)
}

func main() {
    list := make([]worker, 10)
    for n:=0; n

借助接口数组,我可以为 workera 和 `workerb 调用相应的 show 方法。现在我想添加第三个方法调用 add 在两个 worker 之间添加 x```

func (a *workera) add(b workera) {
    a.x += b.x
}

我发现在 list 持有接口而不是结构之前调用以下命令会导致错误

list[0].add(list[2])

我在网上查了一些想法,看到reflect可能有助于解决问题,这是我写的

func callMethodParam(p interface{}, q interface{}) {
    o := reflect.ValueOf(p)
    m := reflect.ValueOf(q)
    args:=[]reflect.Value{reflect.ValueOf(m)}
    o.MethodByName("Add").Call(args)
}

callMethodParam(list[0], list[2])

但这似乎也不起作用。知道如何让它发挥作用吗?谢谢。


正确答案


你的问题的标题听起来像是一个 xy 问题,但幸运的是我们知道 x,不是吗?

此外,代码 list[0].add(list[2]) 无法像示例中显示的那样工作,因为 list 被声明为 worker 的切片,并且 worker 没有 add 方法。 p>

但我们假设确实如此。鉴于 add(b workera) 的实现需要访问 workerax 字段,您可以更改签名以接受提供 x 的接口,并让 workera 实现该接口:

type xprovider interface {
    x() int
}

func (a *workera) x() int {
    return a.x
}

func (a *workera) add(b xprovider) {
    a.x += b.x()
}

然后在调用 a.add 之前,您使用 type assertion 来确保 worker 的特定实例(包含的类型)也实现 xprovider

if xp, ok := list[2].(XProvider); ok {
    list[0].Add(xp)
}

这是更新的 playground

到这里,我们也就讲完了《Golang中如何使用reflect和interface{}实现带参方法的多态?》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>