dynamic动态类型的扩展方法
对于一个动态类型来说,你可以认为它包含任意成员,它们都能通过编译。但到了运行时,到底是否拥有这些成员,就真相大白了。如
dynamic test = 7; Console.Write(test.Name);
编译器无法在编译时知道test的真正类型,因此会使用其运行时的实际类型,而默认对于它的所有调用都是合法的,不会引发任何编译时错误。但它会抛出一个运行时异常。因为在运行时,test为一个int,它不具备Name属性。
在编译时,编译器会根据会生成一些调用所需的上下文环境,如所有已知的静态类型等。但它无法知道在源代码中究竟引入了哪些命名空间。因此,你无法调用动态类型所代表的实际类型的扩展方法,也无法将动态类型作为参数传入扩展方法。如
dynamic size = 5; var numbers = Enumerable.Range(1, 10); var error = numbers.Take(size);
但我们有两种方法可以实现这两点,它们看上去可能有点丑陋,但在你需要的时候却会很有用。这两种方法为:
1. 将动态类型强制转换为已知的符合方法签名的静态类型
2. 以静态方法的形式调用
对于将动态类型作为参数传入扩展方法的情况,你可以这样
dynamic size = 5; var numbers = Enumerable.Range(1, 10); var workaround1 = numbers.Take((int)size); var workaround2 = Enumerable.Take(numbers, size);
对于调用动态类型的扩展方法,可以这样
int size = 5; dynamic numbers = Enumerable.Range(1, 10); var workaround1 = ((IEnumerable<int>)numbers).Take(size); var workaround2 = Enumerable.Take(numbers, size);