Получение имени вызванного метода, выполненного в Func - PullRequest
2 голосов
/ 04 августа 2009

Я хотел бы получить имя метода, который делегируется как Func.

Func<MyObject, object> func = x => x.DoSomeMethod();
string name = ExtractMethodName(func); // should equal "DoSomeMethod"

Как мне этого добиться?

- Для хвастовства -

Заставьте ExtractMethodName также работать с вызовом свойства, чтобы оно возвращало имя свойства в этом экземпляре.

например.

Func<MyObject, object> func = x => x.Property;
string name = ExtractMethodName(func); // should equal "Property"

Ответы [ 3 ]

11 голосов
/ 04 августа 2009

Смотри, мама! Нет деревьев выражений!

Вот быстрая, грязная и специфичная для реализации версия, которая берет токен метаданных из потока IL базовой лямбды и разрешает его.

private static string ExtractMethodName(Func<MyObject, object> func)
{
    var il = func.Method.GetMethodBody().GetILAsByteArray();

    // first byte is ldarg.0
    // second byte is callvirt
    // next four bytes are the MethodDef token
    var mdToken = (il[5] << 24) | (il[4] << 16) | (il[3] << 8) | il[2];
    var innerMethod = func.Method.Module.ResolveMethod(mdToken);

    // Check to see if this is a property getter and grab property if it is...
    if (innerMethod.IsSpecialName && innerMethod.Name.StartsWith("get_"))
    {
        var prop = (from p in innerMethod.DeclaringType.GetProperties()
                    where p.GetGetMethod() == innerMethod
                    select p).FirstOrDefault();
        if (prop != null)
            return prop.Name;
    }

    return innerMethod.Name;
}
0 голосов
/ 04 августа 2009

Проверьте мой хак ответ здесь:

Почему в C # нет оператора `fieldof` или` methodof`?

Раньше я делал это другим способом, который использовал Func вместо Expression<Func<...>>, но я был гораздо менее доволен результатом. MemberExpression, используемый для обнаружения поля в моем методе fieldof, вернет PropertyInfo при использовании свойства.

Редактировать # 1: Это работает для подмножества проблемы:

Func<object> func = x.DoSomething;
string name = func.Method.Name;

Редактировать # 2: Тот, кто пометил меня, должен уделить секунду, чтобы понять, что здесь происходит. Деревья выражений могут неявно использоваться с лямбда-выражениями и являются наиболее быстрым и надежным способом получения конкретной запрашиваемой информации здесь.

0 голосов
/ 04 августа 2009

Я не думаю, что это возможно в общем случае. Что если бы у вас было:

Func<MyObject, object> func = x => x.DoSomeMethod(x.DoSomeOtherMethod());

Что бы вы ожидали?

При этом вы можете использовать отражение, чтобы открыть объект Func и посмотреть, что он делает внутри, но вы сможете решить его только в определенных случаях.

...