将对象列表转换为List vs IList

时间:2013-12-05 19:23:22

标签: c# casting

刚刚遇到这个:

Func<List<object>> foo = () => new List<object>();
List<string> s = (List<string>)foo();
IList<string> s1 = (IList<string>)foo();

编译器抱怨转换为List(有意义),但对IList没有任何说明。让我想知道为什么会这样?

3 个答案:

答案 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),答案就会澄清为什么这个编译有效,并且还提供了一个满足所提供条件的类的例子。