我在单元测试中有以下代码
public bool TestMethodsOf<T, I>()
{
var impl = typeof(T);
var valid = true;
foreach (var iface in impl.GetInterfaces().Where(i => typeof(I).IsAssignableFrom(i)))
{
var members = iface.GetMethods();
foreach (var member in members)
{
Trace.Write("Checking if method " + iface.Name + "." + member.Name + " is virtual...");
var implMember = impl.GetMethod(member.Name, member.GetParameters().Select(c => c.ParameterType).ToArray());
if (!implMember.IsVirtual)
{
Trace.WriteLine(string.Format("FAILED"));
valid = false;
continue;
}
Trace.WriteLine(string.Format("OK"));
}
}
return valid;
}
我打电话给
Assert.IsTrue(TestMethodsOf<MyView, IMyView>());
我想确保界面中的所有方法都声明为虚拟。原因是因为我正在应用spring.net方面,它只适用于虚方法。
我遇到的问题是implMember.IsVirtual始终为true,即使它们未在声明类型中声明为true。
我的TestMethodsOf逻辑出了什么问题?
干杯
答案 0 :(得分:20)
接口中声明的所有方法都标记为virtual abstract
,并且在类中实现接口方法的所有方法都标记为virtual final
,因此CLR知道它不能直接调用它们 - 它必须在运行时进行vtable查找才能调用正确的实现。接口实现仍然是虚拟的,但你不能覆盖它们,因为它们是最终的。
例如,以下C#定义:
public interface IInterface {
void Method();
}
public class Class : IInterface {
public void Method() {}
}
汇编到以下IL:
.class public interface abstract IInterface {
.method public abstract virtual instance void Method() {}
}
.class public Class extends [mscorlib]System.Object implements IInterface {
.method public specialname rtspecialname instance void .ctor() {}
.method public virtual final instance void Method() {}
}
答案 1 :(得分:3)
我相信当你实现一个接口时,你从接口继承的方法会自动标记为虚拟,所以逻辑很好,你不需要测试。