Получить имя метода, строго типизированного

Подумайте, что у меня есть класс, как показано ниже:

public class Foo
{
    public int Bar { get; set; }

    public int Sum(int a, int b)
    {
        return a + b;
    }

    public int Square(int a)
    {
        return a * a;
    }
}

Все, что вы знаете, что я могу написать метод, который возвращает имя данного свойства:

var name = GetPropertyName<Foo>(f => f.Bar); //returns "Bar"

Метод GetPropertyName может быть легко реализован, как показано ниже:

public static string GetPropertyName<T>(Expression<Func<T, object>> exp)
{
    var body = exp.Body as MemberExpression;
    if (body == null)
    {
        var ubody = (UnaryExpression)exp.Body;
        body = ubody.Operand as MemberExpression;
    }
    return body.Member.Name;
}

Но я хочу получить имя метода так же легко, как имя свойства, как показано ниже:

var name1 = GetMethodName<Foo>(f => f.Sum); //expected to return "Sum"
var name2 = GetMethodName<Foo>(f => f.Square); //expected to return "Square"

Можно ли написать такой метод GetMethodName?

Обратите внимание, что: GetMethodName должно быть независимым от подписи или возвращаемого значения данного метода.

Ответ 1

После некоторых исследований я обнаружил, что Expression<Action<T>> должно быть достаточно, чтобы делать то, что вы хотите.

private string GetMethodName<T>(Expression<Action<T>> method)
{
    return ((MethodCallExpression)method.Body).Method.Name;
}

public void TestMethod()
{
    Foo foo = new Foo();
    Console.WriteLine(GetMethodName<Foo>(x => foo.Square(1)));
    Console.WriteLine(GetMethodName<Foo>(x => foo.Sum(1,1)));
    Console.ReadLine();
}

public class Foo
{
    public int Bar { get; set; }

    public int Sum(int a, int b)
    {
        return a + b;
    }

    public int Square(int a)
    {
        return a * a;
    }
}

Output