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.

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

报告相同问题?

悬赏问题

  • ¥20 给自己本科IT专业毕业的妹m找个实习工作
  • ¥15 用友U8:向一个无法连接的网络尝试了一个套接字操作,如何解决?
  • ¥30 我的代码按理说完成了模型的搭建、训练、验证测试等工作(标签-网络|关键词-变化检测)
  • ¥50 mac mini外接显示器 画质字体模糊
  • ¥15 TLS1.2协议通信解密
  • ¥40 图书信息管理系统程序编写
  • ¥20 Qcustomplot缩小曲线形状问题
  • ¥15 企业资源规划ERP沙盘模拟
  • ¥15 树莓派控制机械臂传输命令报错,显示摄像头不存在
  • ¥15 前端echarts坐标轴问题