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

我有一个枚举, Foo

 public enum Foo { Alpha, Bravo, Charlie } 

如果我尝试从盒装intFoo?的以下演员Foo? ,我得到一个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 ,你不能从一个盒装的int转换成除int?之外的任何类型的可空类int?

顺便说一句,我首先将intobject的原因是我真的试图从int转换为通用参数TValue ,如下所示:

 var x = (TValue)(object)1; 

如果我没有(object) ,它将无法编译。 (有关详细信息,请参阅Eric Lippert撰写的此博客文章 。)

问题

  1. 为什么你可以从盒装int转换为枚举,但不能转换为可以为空的枚举(而不是long也不long? )?

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

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

 enum Foo : byte { ... 

你将无法从盒装int转换为Foo。

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

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

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

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

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

我认为这可以回答你的问题!