使用动态类型作为方法参数时的奇怪行为
我有以下接口是现有项目的一部分。我想用动态对象调用Store(..)函数。但我不想更改接口层次结构(如果可能的话)。
public interface IActualInterface
{
void Store(object entity);
}
public interface IExtendedInterface : IActualInterface
{
//Interface items not important
}
public class Test : IExtendedInterface
{
public void Store(object entity)
{
Console.WriteLine("Storing: " + entity.ToString());
}
}
和以下代码:
IExtendedInterface extendedInterfaceTest = new Test();
IActualInterface actualInterfaceTest = new Test();
Test directTest = new Test();
dynamic employee = new ExpandoObject();
employee.Name = "John Smith";
employee.Age = 33;
employee.Phones = new ExpandoObject();
employee.Phones.Home = "0111 123123";
employee.Phones.Office = "027 321123";
employee.Tags = new List<dynamic>() { 123.4D, 99.54D };
try
{
extendedInterfaceTest .Store(employee);
}
catch (RuntimeBinderException rbEx)
{
Console.WriteLine(rbEx.Message);
}
//Casting as (object) works okay as it's not resolved at runtime
extendedInterfaceTest.Store((object)employee);
//this works because IActualInterface implements 'Store'
actualInterfaceTest.Store(employee);
//this also works okay (directTest : IProxyTest)
directTest.Store(employee);
当我调用extendedInterfaceTest.Store(employee)
时,它会引发运行时绑定程序异常。为什么接口类型在相同的底层类型时有所不同?我可以在IActualInterface
和Type
上调用它,但不是IExtendedInterface
吗?
我知道在使用动态参数调用函数时,解析会在运行时发生,但为什么不同的行为呢?
没有找到相关结果
已邀请:
1 个回复
夏瓤跋棘
一切都很好。但编译器真正生成了什么?让我们看看使用ILdasm。 对于接口:
我们在这里可以看到C#编译器总是为定义方法的接口或类生成调用。
有一个存储方法槽,所以它用于
。
没有,所以
用于通话。
定义了一个新方法Store,使用
IL修饰符,在vtable中为该方法有效地分配一个新槽,因此它直接使用,因为
的类型为
。 对于班级:
对于3种不同的类型,生成相同的调用,因为方法槽在ActualClass上定义。 现在让我们看看如果我们自己编写IL,使用我们想要的类型而不是让C#编译器为我们选择它。我修改了IL看起来像这样: 对于接口:
对于课程:
该程序与ILasm编译良好。但是它无法在运行时传递peverify并崩溃,并出现以下错误: 未处理的异常: System.MissingMethodException:方法 没找到:'空虚 ConsoleApplication38.IExtendedInterface.Store(System.Object的)”。 在 ConsoleApplication38.Program.TestInterfaces() 在 ConsoleApplication38.Program.Main(字符串[] 参数) 如果删除此无效调用,派生类调用可以正常工作而不会出现任何错误。 CLR能够从派生类型调用中解析基本方法。但是,接口在运行时没有真正的表示,并且CLR无法从扩展接口解析方法调用。 理论上,C#编译器可以直接将调用发送到运行时中指定的正确类。如Eric Lippert的博客所示,它可以避免中产阶级调用的问题。但是,如所示,接口不可能实现。 让我们回到DLR。它解析方法的方式与CLR完全相同。我们已经看到CLR无法解决
。 DLR也不能! C#编译器会发出正确的调用这一事实完全隐藏了这一点,所以在使用
时要小心,除非你完全知道它在CLR中是如何工作的。