奇怪的C#动态行为

参见英文答案 > Why does dynamic binding fail when using interface inheritance?2个
在研究C#动态关键字如何工作时,我偶然发现了一些奇怪的行为.它几乎看起来像一个bug,但它可能更有可能是这种行为的原因.

在下面的代码中,有两个调用,一个调用obj1,一个调用obj2,但只有一个调用正确执行.似乎局部变量类型是原因,但也应该可以从IDynamicTarget访问“Hello”,因为它扩展了IDynamicTargetBase.

namespace DynamicTesting
{
    interface IDynamicTargetBase
    {
        string Hello(int a);
    }

    interface IDynamicTarget : IDynamicTargetBase
    {
    }

    class DynamicTarget : IDynamicTarget
    {
        public string Hello(int a)
        {
            return "Hello!";
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            dynamic a = 123;

            IDynamicTargetBase obj1 = new DynamicTarget();
            obj1.Hello(a);  // This works just fine

            IDynamicTarget obj2 = new DynamicTarget();
            obj2.Hello(a); // RuntimeBinderException "No overload for method 'Hello' takes '1' arguments"
        }
    }
}

解决方法

这似乎是一个方法重载解决问题.

只需将动态a = 123更改为int a = 123,您的代码即可运行.另外,如果将方法调用更改为obj2.Hello((int)a);.最后,将变量键入DynamicTarget而不是IDynamicTarget,它也可以工作!

为什么?当您使用动态表达式并且不仅有一个方法重载,invokation具有动态参数时,运行时将无法解析调用哪个重载,因为方法重载决策是基于当提供的参数的类型和顺序时所谓的方法被称为.

我的推测是,当接口也实现其他接口时,运行时重载解析失败,并且运行时似乎理解不能保证第二个接口将定义其他接口之一的重载也会实现,并强制您提供编译时参数的实际类型.

[…] but “Hello” should also be accesible from IDynamicTarget,
because it extends IDynamicTargetBase.

它是可访问的,但运行时无法解决如何提供方法的参数…

相关文章

在要实现单例模式的类当中添加如下代码:实例化的时候:frmC...
1、如果制作圆角窗体,窗体先继承DOTNETBAR的:public parti...
根据网上资料,自己很粗略的实现了一个winform搜索提示,但是...
近期在做DSOFramer这个控件,打算自己弄一个自定义控件来封装...
今天玩了一把WMI,查询了一下电脑的硬件信息,感觉很多代码都...
最近在研究WinWordControl这个控件,因为上级要求在系统里,...