douwang4374 2017-10-06 13:47
浏览 432
已采纳

Golang获得反射调用的结果

I'm making a reflection function call but I'm currently stuck on trying to get the return values.

Example code:

func (c *Client) Send(funcName protoFunction, data interface{}) (interface{}, error) {
    funcList := c.GetProtoFuncList()
    ctx := context.Background()
    f := reflect.ValueOf(funcList[funcName.String()])
    vals := make([]reflect.Value, 2)
    vals[0] = reflect.ValueOf(ctx)
    vals[1] = reflect.ValueOf(data)
    value := f.Call(vals)
    log.Debug(value)
    return nil, nil
}

How do I get the return values from "value" and return them correctly?

Cheers

  • 写回答

1条回答 默认 最新

  • dpa55065 2017-10-06 14:19
    关注

    Value.Call() returns the return values of the function call as a []reflect.Value. You may use the Value.Interface() method to get the value represented by a reflect.Value as an interface{} value. From there, you may use type assertion to obtain values of different types.

    Like in this simplified example:

    var funcList = map[string]interface{}{
        "test": func(ctx context.Context, data interface{}) (interface{}, error) {
            return "test-result-data:" + fmt.Sprint(data), errors.New("test-error")
        },
    }
    
    func Send(funcName string, data interface{}) (interface{}, error) {
        f := reflect.ValueOf(funcList[funcName])
        params := []reflect.Value{
            reflect.ValueOf(context.Background()),
            reflect.ValueOf(data),
        }
        res := f.Call(params)
    
        ret := res[0].Interface()
        var err error
        if v := res[1].Interface(); v != nil {
            err = v.(error)
        }
        return ret, err
    }
    

    Testing it:

    result, err := Send("test", "testData")
    fmt.Println(result, err)
    

    Output:

    test-result-data:testData test-error
    

    But this is unnecessarily complex. You don't need to use reflection to call a function, you may call it directly and return the results directly, like this:

    func Send2(funcName string, data interface{}) (interface{}, error) {
        return funcList[funcName].(func(context.Context, interface{}) (interface{}, error))(
            context.Background(), data,
        )
    }
    

    Testing it:

    result, err = Send2("test", "testData")
    fmt.Println(result, err)
    

    Output is the same. Try the examples on the Go Playground.

    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论

报告相同问题?

悬赏问题

  • ¥100 vue3中融入gRPC-web
  • ¥15 kali环境运行volatility分析android内存文件,缺profile
  • ¥15 写uniapp时遇到的问题
  • ¥15 vs 2008 安装遇到问题
  • ¥15 matlab有限元法求解梁带有若干弹簧质量系统的固有频率
  • ¥15 找一个网络防御专家,外包的
  • ¥100 能不能让两张不同的图片md5值一样,(有尝)
  • ¥15 informer代码训练自己的数据集,改参数怎么改
  • ¥15 请看一下,学校实验要求,我需要具体代码
  • ¥50 pc微信3.6.0.18不能登陆 有偿解决问题