仅获取Type.GetMethods()中具有特定签名的方法

时间:2011-03-01 09:04:43

标签: c#

我想列出具有特定方法签名的所有类型的方法。

例如,如果某个类型有一些公共方法:

public void meth1 (int i);
public void meth2 (int i, string s);
public void meth3 (int i, string s);
public int meth4 (int i, string s);

我想列出所有期望int为first的方法,将字符串作为第二个参数,并返回void。

我该怎么做?

5 个答案:

答案 0 :(得分:12)

您可以使用以下内容:

public static class Extensions
{
    public static IEnumerable<MethodInfo> GetMethodsBySig(this Type type, Type returnType, params Type[] parameterTypes)
    {
        return type.GetMethods().Where((m) =>
        {
            if (m.ReturnType != returnType) return false;
            var parameters = m.GetParameters();
            if ((parameterTypes == null || parameterTypes.Length == 0))
                return parameters.Length == 0;
            if (parameters.Length != parameterTypes.Length)
                return false;
            for (int i = 0; i < parameterTypes.Length; i++)
            {
                if (parameters[i].ParameterType != parameterTypes[i])
                    return false;
            }
            return true;
        });
    }
}

并像这样使用它:

var methods =  this.GetType().GetMethodsBySig(typeof(void), typeof(int), typeof(string));

答案 1 :(得分:7)

type.GetMethods().Where(p =>
                p.GetParameters().Select(q => q.ParameterType).SequenceEqual(new Type[] { typeof(int), typeof(string) }) &&
                p.ReturnType == typeof(void)
            );

答案 2 :(得分:4)

您必须自己检查所有MethodInfo。通过调用MethodInfo.GetParameters(),您将获得ParameterInfo个对象的集合,而这些对象又具有属性ParameterType

返回类型相同:检查ReturnType的{​​{1}}属性。

答案 3 :(得分:1)

鉴于此课程:

public class Foo
{
    public void M1(int i){}
    public void M2(int i, string s){}
    public void M3(int i, string s){}
    public int M4(int i, string s){ return 0; }
}

可以使用一些Reflection和LINQ:

Type t = typeof (Foo);
var theMethods = t.GetMethods().Where(mi =>
                            {
                                var p = mi.GetParameters();
                                if (p.Length != 2)
                                    return false;

                                if (p[0].ParameterType != typeof(int) 
                                     || p[1].ParameterType != typeof(string))
                                    return false;

                                return mi.ReturnType == typeof (void);
                            });

或其他语法(在这种情况下实际上更好)

var theMethods = from mi in t.GetMethods()
                    let p = mi.GetParameters()
                    where p.Length == 2
                        && p[0].ParameterType == typeof (int)
                        && p[1].ParameterType == typeof (string)
                        && mi.ReturnType == typeof (void)
                    select mi;

测试:

foreach (var methodInfo in theMethods)
{
    Console.WriteLine(methodInfo.Name);
}

输出:

M2
M3

答案 4 :(得分:0)

这是使用EvgK的答案改进的扩展方法。

 private static IEnumerable<MethodInfo> GetMethodsBySig(this Type type,
                                                                Type returnType,
                                                                Type customAttributeType,
                                                                bool matchParameterInheritence,
                                                                params Type[] parameterTypes)
        {
            return type.GetMethods().Where((m) =>
            {
                if (m.ReturnType != returnType) return false;

            if ((customAttributeType != null) && (m.GetCustomAttributes(customAttributeType, true).Any() == false))
                return false;

            var parameters = m.GetParameters();

            if ((parameterTypes == null || parameterTypes.Length == 0))
                return parameters.Length == 0;

            if (parameters.Length != parameterTypes.Length)
                return false;

            for (int i = 0; i < parameterTypes.Length; i++)
            {
                if (((parameters[i].ParameterType == parameterTypes[i]) ||
                (matchParameterInheritence && parameterTypes[i].IsAssignableFrom(parameters[i].ParameterType))) == false)
                    return false;
            }

            return true;
        });
    }

一样使用它
var methods = SomeTypeToScan.GetMethodsBySig(
                typeof(SomeReturnType),
                typeof(SomeSpecialAttributeMarkerType),
                true, 
                typeof(SomeParameterType))
                .ToList();