InvalidCastException尝试从盒装int转换为可为空的枚举

时间:2012-09-05 23:44:24

标签: c# generics casting enums nullable

我有一个枚举,Foo

public enum Foo { Alpha, Bravo, Charlie }

如果我从盒装intFoo?尝试以下演员,我会得到InvalidCastException

var x = (Foo?)(object)1;

这导致我进行了一些实验......

var x = (Foo)(object)1; // succeeds
var x = (long)(object)1; // fails
var x = (long?)(object)1; // fails
var x = (long)1; // succeeds
var x = (long?)1; // succeeds
var x = (int)(object)1; // succeeds
var x = (int?)(object)1; // succeeds

这告诉我的是,你可以从盒装int转换为枚举,但不能转换为long,而你无法转换为盒装{{1}除int之外的任何类型的可空类。

顺便说一句,我首先将int?转换为int的原因是我真的试图从object转换为通用参数{{1} },像这样:

int

如果我没有TValue,则无法编译。 (有关详细信息,请参阅此blog post by Eric Lippert。)

问题

  1. 为什么你可以从盒装var x = (TValue)(object)1; 转换为枚举,但不能转换为可以为空的枚举(而不是(object)int)?

  2. 重写long的最简单方法是什么,以便它编译,在运行时工作,并且是高性能的(假设long?在运行时被确定为var x = (TValue)(object)1;

2 个答案:

答案 0 :(得分:5)

要回答第一个问题,只有当盒装值是枚举的基础类型时,才能从盒装值转换为枚举值。如果你宣布

enum Foo : byte { ...

你将无法从boxed int转换为Foo。

要回答第二个问题,请尝试

var x = (TValue)Enum.ToObject(typeof(TValue), 1);

这涉及拳击;如果你需要一个不会打包的解决方案,它会更复杂。

答案 1 :(得分:1)

Nullables不是一些特殊类型的原子类型,而是类型Nullable<T>的简写,这就是为什么你不能将盒装int转换为可以为空的枚举。如果你想创建一个可以为空的枚举,你可以这样做:

var x = new Nullable<Foo>((Foo)1);

我认为这回答了你的问题!