奇怪的C#动态行为

时间:2015-01-25 10:00:16

标签: c# dynamic dynamic-language-runtime

在调查C#动态关键字如何工作时,我偶然发现了一些奇怪的行为。它几乎看起来像一个bug,但它可能更有可能是这种行为的原因。

在下面的代码中,有两个调用,一个调用obj1,一个调用obj2,但只有一个调用正确执行。似乎局部变量类型是原因,但是"你好"也应该可以从IDynamicTarget访问,因为它扩展了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"
        }
    }
}

1 个答案:

答案 0 :(得分:0)

这似乎是一种方法重载解决问题。

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

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

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

  

[...]但是也应该从IDynamicTarget访问“Hello”,   因为它扩展了IDynamicTargetBase。

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