How to properly use .Call in reflect package

前端 未结 2 1772
一向
一向 2021-01-02 17:48

Been having one last issue with my code which involves the .Call function in the reflect package.

So I\'m making a call such as this:

params := \"so         


        
相关标签:
2条回答
  • 2021-01-02 18:34

    From the Value.Call documentation:

    Call calls the function v with the input arguments in. For example, if len(in) == 3, v.Call(in) represents the Go call v(in[0], in[1], in[2]).

    So if you want to call a function with one parameter, in must contain one reflect.Value of the right type, in your case map[string][]string.

    The expression

    in := make([]reflect.Value,0)
    

    creates a slice with length 0. Passing this to Value.Call will result in the panic you receive as you need 1 parameter, not zero.

    The correct call would be:

    m := map[string][]string{"foo": []string{"bar"}}
    
    in := []reflect.Value{reflect.ValueOf(m)}
    
    myMethod.Call(in)
    
    0 讨论(0)
  • 2021-01-02 18:47

    The call is trying to pass zero parameters to a controller that expects one param (in is an empty slice). You need to do something more like in := []reflect.Value{reflect.ValueOf(params)}.

    You could also call .Interface() once you've found the method, then use type assertion to get a func you can call directly:

    // get a reflect.Value for the method
    methodVal := reflect.ValueOf(&controller_ref).MethodByName(action_name)
    // turn that into an interface{}
    methodIface := methodVal.Interface()
    // turn that into a function that has the expected signature
    method := methodIface.(func(map[string][]string) map[string]string)
    // call the method directly
    res := method(params)
    

    (Then you could even cache method in a map keyed by method name, so you wouldn't have to do reflect operations next call. But you don't have to do that for it to work.)

    0 讨论(0)
提交回复
热议问题