C# 为什么我不能将 int 拆箱为小数?
声明:本页面是StackOverFlow热门问题的中英对照翻译,遵循CC BY-SA 4.0协议,如果您需要使用它,必须同样遵循CC BY-SA许可,注明原文地址和作者信息,同时你必须将它归于原作者(不是我):StackOverFlow
原文地址: http://stackoverflow.com/questions/1085097/
Warning: these are provided under cc-by-sa 4.0 license. You are free to use/share it, But you must attribute it to the original authors (not me):
StackOverFlow
Why can't I unbox an int as a decimal?
提问by mezoid
I have an IDataRecord reader
that I'm retrieving a decimal from as follows:
我有一个IDataRecord reader
我正在从中检索小数的信息,如下所示:
decimal d = (decimal)reader[0];
For some reason this throws an invalid cast exception saying that the "Specified cast is not valid."
出于某种原因,这会引发无效的强制转换异常,指出“指定的强制转换无效”。
When I do reader[0].GetType()
it tells me that it is an Int32. As far as I know, this shouldn't be a problem....
当我这样做时,reader[0].GetType()
它告诉我它是一个 Int32。据我所知,这应该不是问题......
I've tested this out by this snippet which works just fine.
我已经通过这个片段测试了这一点,它工作得很好。
int i = 3750;
decimal d = (decimal)i;
This has left me scratching my head wondering why it is failing to unbox the int contained in the reader as a decimal.
这让我摸不着头脑,想知道为什么它无法将读取器中包含的 int 拆箱为十进制。
Does anyone know why this might be occurring? Is there something subtle I'm missing?
有谁知道为什么会发生这种情况?我错过了一些微妙的东西吗?
采纳答案by Mehrdad Afshari
You can only unbox a value type to its original type (and the nullable version of that type).
您只能将值类型拆箱为其原始类型(以及该类型的可为空版本)。
By the way, this is valid (just a shorthand for your two line version):
顺便说一句,这是有效的(只是您的两行版本的简写):
object i = 4;
decimal d = (decimal)(int)i; // works even w/o decimal as it's a widening conversion
For the reason behind this read this Eric Lippert's blog entry: Representation and Identity
对于这背后的原因,请阅读Eric Lippert 的博客文章:Representation and Identity
Personally, I categorize things done by cast syntax into four different types of operation (they all have different IL instructions):
就我个人而言,我将 cast 语法所做的事情分为四种不同类型的操作(它们都有不同的 IL 指令):
- Boxing (
box
IL instruction) and unboxing (unbox
IL instruction) - Casting through the inhertiance hierarchy (like
dynamic_cast<Type>
in C++, usescastclass
IL instruction to verify) - Casting between primitive types (like
static_cast<Type>
in C++, there are plenty of IL instructions for different types of casts between primitive types) - Calling user defined conversion operators (at the IL level they are just method calls to the appropriate
op_XXX
method).
- 装箱(
box
IL指令)和拆箱(unbox
IL指令) - 通过继承层次结构进行转换(如
dynamic_cast<Type>
在 C++ 中,使用castclass
IL 指令进行验证) - 原始类型之间的转换(就像
static_cast<Type>
在 C++ 中,有很多 IL 指令用于原始类型之间的不同类型的转换) - 调用用户定义的转换运算符(在 IL 级别,它们只是对适当
op_XXX
方法的方法调用)。
回答by Guffa
There is no problem in casting an int
to decimal
, but when you are unboxing an object you have to use the exact type that the object contains.
将int
to强制转换没有问题decimal
,但是当您拆箱对象时,您必须使用该对象包含的确切类型。
To unbox the int
value into a decimal
value, you first unbox it as an int, then cast it to decimal:
要将int
值拆箱为decimal
值,首先将其拆箱为 int,然后将其转换为十进制:
decimal d = (decimal)(int)reader[0];
The IDataRecord interface also has methods for unboxing the value:
IDataRecord 接口还具有对值进行拆箱的方法:
decimal d = (decimal)reader.GetInt32(0);
回答by user276648
Mehrdad Afshari said it:
Mehrdad Afshari 说:
You can only unbox a value type to its original type (and the nullable version of that type).
您只能将值类型拆箱为其原始类型(以及该类型的可为空版本)。
The thing to realize is that there is a difference between casting and unboxing. jerryjvl had an excellent remark
要意识到的是,cast 和 unboxing 之间存在差异。jerryjvl 发表了精彩的评论
In a sense it's a shame that unboxing and casting syntactically look identical, since they are very different operations.
从某种意义上说,拆箱和强制转换在语法上看起来相同是一种耻辱,因为它们是非常不同的操作。
Casting:
铸件:
int i = 3750; // Declares a normal int
decimal d = (decimal)i; // Casts an int into a decimal > OK
Boxing/Unboxing:
装箱/拆箱:
object i = 3750; // Boxes an int ("3750" is similar to "(int)3750")
decimal d = (decimal)i; // Unboxes the boxed int into a decimal > KO, can only unbox it into a int or int?
回答by Sagar
Here is a simple solution. It takes care of unboxing and then casting to decimal. Worked fine for me.
这是一个简单的解决方案。它负责拆箱,然后转换为十进制。对我来说效果很好。
decimal d = Convert.ToDecimal(reader[0]); // reader[0] is int