【问题标题】:Calling interface method after reflection & type switching in golang在golang中反射和类型切换后调用接口方法
【发布时间】:2020-08-04 10:48:44
【问题描述】:

我有结构

type ChartOpts struct {
    Name              mypakage.MyType
    Repo              mypakage.MyType
}

我想迭代其字段,如果满足类型断言,则调用该(断言)类型的方法。

func (chartOpts ChartOpts) BindFlags(cobCom *cobra.Command) {
    fields := reflect.TypeOf(chartOpts)
    values := reflect.ValueOf(chartOpts)
    num := fields.NumField()
    fmt.Println(fields, values, num)
    for i := 0; i < num; i++ {
        field := fields.Field(i)
        switch v := field.Type.(type) {
        case mypackage.MyType:
            field.BindPersistentFlag(cobCom)
        default:
            log.Fatal(ErrUnhandledType)
        }
    }
    

}

上面的代码没有编译,出现这个错误:

field.BindPersistentFlag undefined (type reflect.StructField has no field or method BindPersistentFlag)

为什么?

【问题讨论】:

  • field 是 reflect.StructField 类型,它没有 BindPersistentFlag 方法,原因很明显。

标签: go methods struct reflection type-switch


【解决方案1】:

fieldreflect.StructField 类型,它不是字段的值,而是它的结构字段描述符。

您需要在字段的值上进行类型切换:

values.Field(i)

注意,上面是reflect.Value包裹的字段值,所以用Value.Interface()提取值:

for i := 0; i < num; i++ {
    switch v := values.Field(i).Interface().(type) {
    case mypackage.MyType:
        v.BindPersistentFlag(cobCom)
    default:
        log.Fatal(ErrUnhandledType)
    }
}

然后在v 上调用BindPersistentFlag(),因为这将是case 内的mypackage.MyType 类型。

【讨论】:

    【解决方案2】:

    您需要使用该值并从反射中恢复一个接口:

    values.Field(i).Interface().(Type).Method()
    

    在您的情况下,切换界面会更简单:

    switch v := values.Field(i).Interface().(type) {
    case Type:
        v.Method()
    ...
    

    【讨论】:

      猜你喜欢
      • 2017-12-11
      • 2020-05-15
      • 1970-01-01
      • 1970-01-01
      • 2013-10-04
      • 2016-11-18
      • 2014-01-08
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多