gpt4 book ai didi

Go 将字段名称反射(reflect)到特定接口(interface)

转载 作者:数据小太阳 更新时间:2023-10-29 03:09:30 25 4
gpt4 key购买 nike

我有一个结构类型,其中有许多字段都实现了 Renderer 接口(interface)。字段类型实现了与指针接收器的接口(interface)。我想要一个接受字段名称并调用该字段上的 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() 时,它似乎是底层类型。如果我切换到非指针接收器,那么第一个函数就可以工作。我找到了 reflect value Interface and pointer receiver这似乎几乎正是我要问的问题,问题得到了回答,但如果我真的调用了 playground 链接中提供的 isZeroer 方法,它总是错误的,所以它实际上似乎没有回答问题。

好像Addr是关键,因为它特别提到了指针接收器,但我正在努力将其强制转换回接口(interface)。

最佳答案

使用这段代码:

func renderField(name string, f *First) {
structVal := reflect.ValueOf(f).Elem()
field := structVal.FieldByName(name).Addr().Interface()
if renderer, ok := field.(Renderer); ok {
renderer.Render()
}
}

重点是改:

structVal := reflect.ValueOf(*f)

到:

structVal := reflect.ValueOf(f).Elem()

问题中使用的语句创建了一个不可寻址的结构值。不可寻址结构中的字段也是不可寻址的,因此无法访问字段上的指针接收器。

此答案中使用的语句创建了一个可寻址的结构值。

关于Go 将字段名称反射(reflect)到特定接口(interface),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54191322/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com