奇怪的C#动态行为

前端之家收集整理的这篇文章主要介绍了奇怪的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.

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

猜你在找的C#相关文章