猿问

Go 将字段名称反映到特定接口

我有一个包含许多字段的结构类型,这些字段都实现了 Renderer 接口。字段类型实现了与指针接收器的接口。我想要一个接受字段名称并调用该字段上的 Render 方法的函数。我能够找到该字段并获得大量有关它的信息,但是由于指针接收器的原因,进行类型断言似乎让我很头疼。这是一些显示我的问题的代码:

package main


import (

    "fmt"

    "reflect"

)


type Renderer interface {

    Render()

}


type First struct {

    ExampleField Field

}


type Field []int


func (f *Field) Render() {

    fmt.Println("Hello from first")

}


func main() {

    f := First{

        Field{1, 2, 3},

    }

    f.ExampleField.Render()

    renderField("ExampleField", &f)

    renderField2("ExampleField", &f)

}


func renderField(field string, f *First) {

    structVal := reflect.ValueOf(*f)

    renderType := reflect.TypeOf((*Renderer)(nil)).Elem()

    fieldToRender := structVal.FieldByName(field)

    fieldPtr := reflect.PtrTo(fieldToRender.Type())

    fmt.Printf("Implements? %v\n", fieldPtr.Implements(renderType))

    fmt.Printf("Addressable? %v\n", fieldToRender.CanAddr())

    fieldInter := fieldToRender.Interface()

    if renderer, ok := fieldInter.(Renderer); ok {

        // Pointer receiver so this never gets called

        fmt.Print("Able to cast")

        renderer.Render()

    }

}


func renderField2(field string, f *First) {

    structVal := reflect.ValueOf(*f)

    fieldToRender := structVal.FieldByName(field)

    vp := reflect.New(reflect.TypeOf(fieldToRender))

    vp.Elem().Set(reflect.ValueOf(fieldToRender))

    vpAddr := vp.Elem().Addr()

    typeVal := vpAddr.Interface()

    fmt.Println(typeVal) // <main.Field Value>⏎

    renderer := typeVal.(Renderer)

    renderer.Render()

    // interface conversion: *reflect.Value is not main.Renderer: missing method Render

}

renderField2 似乎让我接近但 Addr() 给了我一个 *Reflect.Value 并且当我调用 Interface() 时它似乎是底层类型。如果我切换到非指针接收器,那么第一个函数就可以工作。我发现反射值接口和指针接收器这似乎几乎正是我要问的,问题得到了回答,但如果我实际上调用了操场链接中显示的 isZeroer 方法,它总是错误的,所以它实际上似乎没有回答这个问题。

似乎Addr是关键,因为它特别提到了指针接收器,但我正在努力将其强制返回到接口中。


眼眸繁星
浏览 102回答 1
1回答

喵喵时光机

使用此代码:func renderField(name string, f *First) {&nbsp; &nbsp; structVal := reflect.ValueOf(f).Elem()&nbsp; &nbsp; field := structVal.FieldByName(name).Addr().Interface()&nbsp; &nbsp; if renderer, ok := field.(Renderer); ok {&nbsp; &nbsp; &nbsp; &nbsp; renderer.Render()&nbsp; &nbsp; }}关键是要改:structVal := reflect.ValueOf(*f)到:structVal := reflect.ValueOf(f).Elem()问题中使用的语句创建了一个不可寻址的结构值。不可寻址结构中的字段也不可寻址,因此无法访问字段上的指针接收器。此答案中使用的语句创建了一个可寻址的结构值。
随时随地看视频慕课网APP

相关分类

Go
我要回答