我今天遇到了一个问题,我不完全确定为什么它不起作用。
以下代码示例将崩溃:
static void Main(string[] args)
{
int i32 = 10;
object obj = i32;
long i64 = (long)obj;
}
这将导致InvalidCastException。为什么这不起作用? C#不够聪明,不知道该对象实际上是int类型吗?
我已经提出了一个解决方法,但我很好奇为什么上面的代码示例首先不起作用。
谢谢, 添
答案 0 :(得分:9)
从盒装Int32到Int64没有可用的强制转换。
对int
进行中间转换应该有效,因为编译器愿意生成这个:
// verify obj is a boxed int, unbox it, and perform the *statically*
// known steps necessary to convert an int to a long
long i64 = (long) ((int)obj);
但不是(假设)这个:
// Find out what type obj *actually* is at run-time and perform
// the-known-only-at-run-time steps necessary to produce
// a long from it, involving *type-specific* IL instructions
long i64 = (long)obj;
这是Eric Lippert对此的blog post。
答案 1 :(得分:3)
查看Eric Lippert的this blog post了解详情。
它的要点是编译器弄清楚(通过反复试验,因为object
可以是任何东西)已经装箱的类型以及它是否可以是安全演员。
答案 2 :(得分:1)
您的意思是编译器还是运行时?
运行时足够智能,因此会抛出InvalidCastException
。然而,编译器无法确定您的对象是什么类型,因为您装箱了你的int。
装箱和拆箱启用值类型 被视为对象。拳击a value类型将其打包在一个 Object引用的实例 类型。
因此,由于它被装箱作为对象,编译器不会抱怨它。
在此处查看有关装箱和拆箱的更多信息:
http://msdn.microsoft.com/en-us/library/yz2be5wk%28VS.80%29.aspx