刚刚遇到这个:
Func<List<object>> foo = () => new List<object>();
List<string> s = (List<string>)foo();
IList<string> s1 = (IList<string>)foo();
编译器抱怨转换为List(有意义),但对IList没有任何说明。让我想知道为什么会这样?
答案 0 :(得分:13)
编译器知道List<X>
不能是List<Y>
因此它会产生编译错误。
但是,如果List<X>
实际上是某个也实现IList<Y>
的派生类,则第二次强制转换可能会成功。
如果类型都不是接口,或者一个类型是不相关的接口而另一个类型是密封的(或结构),则只会从转换中获得编译时错误。
引用规范(§6.4.2)
显式引用转换为:
- 从对象和动态到任何其他引用类型。
- 从任何类型S到任何类型类型T,只要S是T的基类。
- 从任何类型S到任何接口类型T,如果S未密封且提供S未实现T。
- 从任何接口类型S到任何类型类型T,如果T未密封或提供T实现S。
- 从任何接口类型S到任何接口类型T,只要S不是从T。
派生的- [剪断]
(强调补充)
provided...
子句排除了实际隐含的转化。
答案 1 :(得分:3)
一个已知为List<object>
的对象可能会实现IList<string>
以及IList<object>
,因此强制转换可能会成功。在这种情况下,它不能因为我们知道该语句只是new List<object>()
,但编译器不考虑这一点。你可能已经扩展List<T>
并实现了另一个,例如
// not recommended, but second cast works
public class MyWeirdList : List<object>, IList<string>
已知为List<object>
的对象也不可能是List<string>
,因为您只能从单一类型继承。
public class MyWeirdList : List<object>, List<string> // compiler error
如果密封List<T>
,则两个强制转换都无效,因为编译器会确定该类无法实现IList<string>
。您可以通过使用此类来尝试此操作:
public sealed class SealedList<T> : List<T> { }
答案 2 :(得分:1)
第一行在编译时失败,第二行给出“无法转换类型的对象'System.Collections.Generic.List 1[System.Object]' to type 'System.Collections.Generic.IList
1 [System.String]'。”运行期间的异常。
如果你看一下这个问题(Cast IList<string> to IList<object> fails at runtime),答案就会澄清为什么这个编译有效,并且还提供了一个满足所提供条件的类的例子。