Reflection говорит, что метод интерфейса является виртуальным в реализованном типе, а когда нет? - PullRequest
12 голосов
/ 25 января 2011

У меня есть следующий код в модульном тесте

    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, даже если они не объявлены кактак в объявлении типа.

Что не так с моей логикой TestMethodsOf?

Приветствия

Ответы [ 2 ]

20 голосов
/ 25 января 2011

Все методы, объявленные в интерфейсе, помечены как 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() {}
}
3 голосов
/ 25 января 2011

Я полагаю, что при реализации интерфейса методы, которые вы наследуете от интерфейса, автоматически помечаются как виртуальные, поэтому с логикой все в порядке, и вам не нужен тест.

...