获取方法的所有派生方法

时间:2019-06-14 14:58:01

标签: c# reflection

给定C#中的方法,我需要在子类中找到所有直接或间接覆盖该方法的方法。

例如,给定以下类架构

abstract class A { public abstract void m(); }
class B : A { public override void m(){}; }
class C : A { public override void m(){}; }
class D : C { public override void m(){}; }
class E : C { public override void m(){}; }
class F : E { public override void m(){}; }
class G : C {  }
class H : G { public override void m(){}; }

然后C.m的派生方法将是来自D,E,F,H类的方法。

我想出了以下解决方案,该解决方案似乎可以正常工作,但发现它有些麻烦。我怀疑有一个聪明的方法可以做到这一点。有什么想法吗?

    public static IEnumerable<MethodInfo> DerivedMethods(this MethodInfo mi)
    {
        return mi.DeclaringType.Assembly.GetTypes()
            .Where(klass => klass.IsSubclassOf(mi.DeclaringType))
            .Select(subclass => subclass.GetMethods(
                    BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance)
                .SingleOrDefault(m => m.GetBaseDefinition() == mi.GetBaseDefinition()))
            .Where(x => x != null);
    }

1 个答案:

答案 0 :(得分:-1)

使其更紧凑的一种方法是改用LINQ关键字。

SingleOrDefault调用有点多余。每个类的每个方法最多只能有一个重写。

此外,我注意到您的代码仅适用于基本接口的工作方式有所不同,不确定您是否也想实现吗?

public static IEnumerable<MethodInfo> DerivedMethods(this MethodInfo mi)
{
    Type baseType = mi.DeclaringType;
    IEnumerable<Type> subClasses = baseType.Assembly.GetTypes().Where((klass) => klass != baseType && baseType.IsAssignableFrom(klass));
    if (baseType.IsInterface) {
        return from klass in subClasses
               where !klass.IsInterface
               let map = klass.GetInterfaceMap(baseType)
               select map.TargetMethods[Array.IndexOf(map.InterfaceMethods, mi)];
    }
    else {
        return from klass in subClasses
               where !klass.IsAbstract // include this only if you want instantiable classes
               from m in klass.GetMethods(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance)
               where m.GetBaseDefinition() == mi.GetBaseDefinition()
               select m;
    }
}

如果您想将搜索范围扩展到解决方案中的所有程序集,则可以将subClasses更改为:

    IEnumerable<Type> subClasses = from asm in AppDomain.CurrentDomain.GetAssemblies()
                                   from klass in asm.GetTypes()
                                   where klass != baseType && baseType.IsAssignableFrom(klass)
                                   select klass;