给定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);
}
答案 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;