以下引发InvalidCastException

IEnumerable<int> list = new List<int>() { 1 };
IEnumerable<long> castedList = list.Cast<long>();
Console.WriteLine(castedList.First());

为什么?

我正在使用Visual Studio 2008 SP1。

最佳答案

太奇怪了!有一个博客文章here描述了.net 3.5和.NET 3.5 SP1之间Cast<T>()的行为是如何变化的,但是它仍然没有解释InvalidCastException,如果您这样重写代码,甚至会得到InvalidCastException:

var list = new[] { 1 };
var castedList = from long l in list select l;
Console.WriteLine(castedList.First());

显然,您可以自己进行转换来解决它
var castedList = list.Select(i => (long)i);

这行得通,但首先并不能解释错误。我尝试将列表强制转换为short和float,并且引发了相同的异常。

编辑

该博客文章确实解释了为什么它不起作用!
Cast<T>()IEnumerable而不是IEnumerable<T>的扩展方法。这意味着,当每个值到达要转换的位置时,它已经被装箱回到System.Object中。本质上,它正在尝试这样做:
int i = 1;
object o = i;
long l = (long)o;

这段代码抛出您得到的InvalidCastException。如果您尝试将int直接转换为long类型,就可以了,但是将boxed int重新转换为long类型是行不通的。

当然很奇怪!

10-06 12:05