Retrieving the name of the invoked method executed in a Func

前端 未结 3 1661
时光取名叫无心
时光取名叫无心 2020-12-18 15:03

I would like to get the name of the method that is being delegated as a Func.

Func func = x => x.DoSomeMethod();
string name = Ext         


        
相关标签:
3条回答
  • 2020-12-18 15:28

    Check out my hack answer here:

    Why is there not a `fieldof` or `methodof` operator in C#?

    In the past I did it another way that used Func instead of Expression<Func<...>>, but I was much less pleased with the result. The MemberExpression used to detect the field in my fieldof method will return a PropertyInfo when a property is used.

    Edit #1: This works for a subset of the problem:

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

    Edit #2: Whoever marked me down should take a second to realize what's going on here. The expression trees can be implicitly used with lambda expressions and are the fastest, most reliable way to get the specific requested information here.

    0 讨论(0)
  • 2020-12-18 15:29

    I don't think this is possible in the general case. What if you had:

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

    What would you expect?

    That being said, you could use reflection to open up the Func object and see what it does inside, but you'll only be able to solve it for certain cases.

    0 讨论(0)
  • 2020-12-18 15:35

    Look Ma! No expression trees!

    Here's a quick, dirty and implementation-specific version that grabs the metadata token from the IL stream of the underlying lambda and resolves it.

    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 讨论(0)
提交回复
热议问题