Получение делегата из methodinfo

У меня есть выпадающий список, который заполняется путем проверки методов класса, включая те, которые соответствуют определенной сигнатуре. Проблема в том, что вы берете выбранный элемент из списка и заставляете делегата вызывать этот метод в классе. Первый метод работает, но я не могу понять часть второго.

Например,

public delegate void MyDelegate(MyState state);

public static MyDelegate GetMyDelegateFromString(string methodName)
{
    switch (methodName)
    {
        case "CallMethodOne":
            return MyFunctionsClass.CallMethodOne;
        case "CallMethodTwo":
            return MyFunctionsClass.CallMethodTwo;
        default:
            return MyFunctionsClass.CallMethodOne;
    }
}

public static MyDelegate GetMyDelegateFromStringReflection(string methodName)
{
    MyDelegate function = MyFunctionsClass.CallMethodOne;

    Type inf = typeof(MyFunctionsClass);
    foreach (var method in inf.GetMethods())
    {
        if (method.Name == methodName)
        {
            //function = method;
            //how do I get the function to call?
        }
    }

    return function;
}

Как заставить закомментированный раздел второго метода работать? Как мне разыграть MethodInfo в делегат?

Спасибо!

Изменить: вот рабочее решение.

public static MyDelegate GetMyDelegateFromStringReflection(string methodName)
{
    MyDelegate function = MyFunctionsClass.CallMethodOne;

    Type inf = typeof(MyFunctionsClass);
    foreach (var method in inf.GetMethods())
    {
        if (method.Name == methodName)
        {
            function = (MyDelegate)Delegate.CreateDelegate(typeof(MyDelegate), method);
        }
    }

    return function;
}

3 ответа

Решение

Вам нужно будет вызвать некоторую форму Delegate.CreateDelegate (), в зависимости от того, является ли рассматриваемый метод статическим или методом экземпляра.

private static Delegate CreateDelegate(this MethodInfo methodInfo, object target) {
    Func<Type[], Type> getType;
    var isAction = methodInfo.ReturnType.Equals((typeof(void)));
    var types = methodInfo.GetParameters().Select(p => p.ParameterType);

    if (isAction) {
        getType = Expression.GetActionType;
    }
    else {
        getType = Expression.GetFuncType;
        types = types.Concat(new[] { methodInfo.ReturnType });
    }

    if (methodInfo.IsStatic) {
        return Delegate.CreateDelegate(getType(types.ToArray()), methodInfo);
    }

    return Delegate.CreateDelegate(getType(types.ToArray()), target, methodInfo.Name);
}

Вот более простая версия решения Саги, использующаяExpression.GetDelegateType()вместо того, чтобы вручную определять, требуется ли действие или функция :

      public static Delegate CreateDelegate(this MethodInfo methodInfo, object target)
{
    var parmTypes = methodInfo.GetParameters().Select(parm => parm.ParameterType);
    var parmAndReturnTypes = parmTypes.Append(methodInfo.ReturnType).ToArray();
    var delegateType = Expression.GetDelegateType(parmAndReturnTypes);

    if (methodInfo.IsStatic)
        return methodInfo.CreateDelegate(delegateType);
    return methodInfo.CreateDelegate(delegateType, target);
}
Другие вопросы по тегам