奇怪的C#动态行为

LaZ*_*aZe 7 c# dynamic-language-runtime dynamic

在研究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"
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

Mat*_*zer 0

看起来这是一个方法重载解析问题。

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

为什么?当您使用动态表达式并且存在多个调用具有动态参数的方法重载时,运行时将无法解析要调用的重载,因为方法重载解析基于以下情况下提供的参数的类型和顺序:所谓的方法就被调用了。

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

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

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