C# 类型推断得到错误的类型
我创建了以下属性,如果在 ViewState[TOTAL_RECORD_COUNT]
为 null
时访问 getter,则会引发 InvalidCastException
。
public long TotalRecordCount
{
get { return (long)(ViewState[TOTAL_RECORD_COUNT] ?? -1); }
set { ViewState[TOTAL_RECORD_COUNT] = value; }
}
我的想法是,它错误地尝试将 ViewState[TOTAL_RECORD_COUNT]
中的对象拆箱为 int
,但失败了,因为它包含 long
,但是我认为这个逻辑可能存在缺陷。我将把它作为练习留给读者来指出这个缺陷。
从那以后,我改变了该属性,以读取
public long TotalRecordCount
{
get { return (long?)ViewState[TOTAL_RECORD_COUNT] ?? -1; }
set { ViewState[TOTAL_RECORD_COUNT] = value; }
}
它的效果。尽管如此,我仍然想知道我的原始版本出了什么问题...... StackOverflow 来救援吗?
请注意,如果我尝试在立即窗口中执行 (long)(ViewState[TOTAL_RECORD_COUNT] ?? -1)
,我会收到错误消息 Cannot unbox 'ViewState[TOTAL_RECORD_COUNT] ?? -1' 作为“long”
,如果我执行 (ViewState[TOTAL_RECORD_COUNT] ?? -1).GetType().Name
我得到 Int32
。我可以执行 (long)-1
并最终得到 -1 作为 Int64
...那么怎么了?
I created the following property, which threw an InvalidCastException
if the getter was accessed when ViewState[TOTAL_RECORD_COUNT]
was null
.
public long TotalRecordCount
{
get { return (long)(ViewState[TOTAL_RECORD_COUNT] ?? -1); }
set { ViewState[TOTAL_RECORD_COUNT] = value; }
}
My thought is that it incorrectly attempted to unbox the object in ViewState[TOTAL_RECORD_COUNT]
to an int
, which failed because it contained a long
, but I think there might be a flaw in that logic. I will leave it as an exercise to the reader to point out that flaw.
I have since changed that property to read
public long TotalRecordCount
{
get { return (long?)ViewState[TOTAL_RECORD_COUNT] ?? -1; }
set { ViewState[TOTAL_RECORD_COUNT] = value; }
}
which works just swell. Still, I am left wondering what was wrong with my original version... StackOverflow to the rescue?
Note that if i try to execute (long)(ViewState[TOTAL_RECORD_COUNT] ?? -1)
in the Immediate Window, I get the error message Cannot unbox 'ViewState[TOTAL_RECORD_COUNT] ?? -1' as a 'long'
and if I execute (ViewState[TOTAL_RECORD_COUNT] ?? -1).GetType().Name
I get Int32
. I can execute (long)-1
and end up with -1 as an Int64
...so what's up?
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(5)
ViewState
索引器的返回类型是Object
(我假设您在这里指的是 ASP.NET 视图状态)。现在考虑一下当编译器看到这个(相当于你的代码)时它必须做什么:它必须推断出表达式
o ?? 的结果类型。 -1 不知何故。左边是一个
对象
,右边是一个int
。显然,该表达式最通用的类型也是object
。但是,这意味着如果它实际上最终使用了-1
(因为o
为 null),则必须将其转换为object
- 对于int
来说,这意味着拳击。所以
x
是object
类型,它可能包含一个int
(也许还有一些其他整数类型 - 我们不知道什么是例如,在您的视图状态中,它可能是短
)。现在你可以写:由于
x
是object
,所以这是拆箱。但是,您只能将值类型拆箱为完全相同的类型(唯一的例外是您可以用有符号类型替换等效的无符号类型,并用枚举替换其基础基类型)。也就是说,您无法将int
拆箱为long
。重现此问题的一种更简单的方法(无需“额外”代码)是:它也会抛出
InvalidCastException
,并且出于完全相同的原因。The return type of
ViewState
indexer isObject
(I assume you mean ASP.NET viewstate here). Now consider what the compiler has to do when it sees this (which is equivalent to your code):It has to deduce the result type of expression
o ?? -1
somehow. On the left it sees anobject
, on the right is anint
. Clearly, the most general type for this expression is alsoobject
. However, this means that if it actually ends up using that-1
(becauseo
was null), it will have to convert it toobject
- and for anint
, this means boxing.So
x
is of typeobject
, and it may contain anint
(and maybe also some other integral type - we don't know what is in your viewstate, it could beshort
, for example). Now you write:Since
x
isobject
, this is unboxing. However, you can only unbox value types into exact same type (with the only exception being that you can substitute a signed type for an equivalent unsigned type, and enum for its underlying base type). That is, you cannot unboxint
intolong
. A far simpler way to repro this, with no "extra" code, would be:Which also throws
InvalidCastException
, and for exact same reason.演员阵容必须只有一步。
表达式
由于无法将包含 Int32 的对象拆箱为 long,因此需要首先将其拆箱为 Int32,然后将其转换为 long。
A cast has to be one step only.
The expression
<object> ?? <int>
will produce another object, and when the first value is null, ie.ViewState[TOTAL_RECORD_COUNT]
is null, then the resulting value will be an object, with a boxed Int32 in it.Since you cannot unbox an object containing an Int32 to a long, you need to first unbox it to an Int32, and then cast it to a long.
问题不在于
ViewState[TOTAL_RECORD_COUNT]
的拆箱,问题在于 -1 的装箱和拆箱。你正在使用?? “object”和“int”上的运算符。结果类型是“对象”。这意味着当视图状态中不存在该字段时,-1 将被装箱(作为 int)。
然后,当您的程序尝试将 (int)-1 作为 long 拆箱时,它会崩溃。
The problems is not the unboxing of the
ViewState[TOTAL_RECORD_COUNT]
, the problem is the boxing and unboxing of the -1.You are using the ?? operator on "object" and "int". The resulting type is "object". This means the -1 will be boxed (as int) when the field does not exist in the view state.
Then your program crashes later when it tries to unbox the (int)-1 as a long.
原文中,如果你把它分解,你正在做:
在你的 运算符 (??) 专门设计用于:
在您的情况下,您使用它来处理 System.Object,因此它将将“-1”视为 Int32,并将其装箱到新的 System.Object 中。然后,它尝试将 Int32 拆箱为 long,但失败,因为转换无法在一步中拆箱并更改类型。
您可以通过使用 L 后缀指定 -1 为 long 来轻松解决此问题:
In your original, if you break it down, you were doing:
The null-coalescing operator (??) is specifially designed to:
In your case, you're using it to handle a System.Object, so it's going to take your "-1", treat it as an Int32, and box it into a new System.Object. Then, it tries to unbox the Int32 into a long, which fails, since the cast cannot unbox and change the type in a single step.
You can solve this easily by specifying that your -1 is a long by using the L suffix:
Int64 是一种值类型,因此转换
null
到值类型总是会抛出异常 (NullReferenceException
)。将 Int32 转换为 Int64 将成功,并且不会抛出InvalidCastException
。Int64 is a value type, so casting
null
to a value type will always throw an exception (NullReferenceException
). And casting an Int32 to Int64 will succeed and will not throw anInvalidCastException
.