反思说接口方法在实现类型中是虚拟的,当它们不是时?

时间:2011-01-25 11:58:17

标签: c# reflection virtual spring.net

我在单元测试中有以下代码

    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逻辑出了什么问题?

干杯

2 个答案:

答案 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)

我相信当你实现一个接口时,你从接口继承的方法会自动标记为虚拟,所以逻辑很好,你不需要测试。