C#7.3最终支持此功能!
以下代码段(来自dotnet示例)演示了如何:
public static Dictionary<int, string> EnumNamedValues<T>() where T : System.Enum
{
var result = new Dictionary<int, string>();
var values = Enum.GetValues(typeof(T));
foreach (int item in values)
result.Add(item, Enum.GetName(typeof(T), item));
return result;
}
确保将C#项目中的语言版本设置为7.3版。
原始答案如下:
我玩游戏迟到了,但是我把它当作挑战,看看如何完成。在C#(或VB.NET,但向下滚动到F#)中是不可能的,但在MSIL中是可能的。我写的这么少。。。
// license: http://www.apache.org/licenses/LICENSE-2.0.html
.assembly MyThing{}
.class public abstract sealed MyThing.Thing
extends [mscorlib]System.Object
{
.method public static !!T GetEnumFromString<valuetype .ctor ([mscorlib]System.Enum) T>(string strValue,
!!T defaultValue) cil managed
{
.maxstack 2
.locals init ([0] !!T temp,
[1] !!T return_value,
[2] class [mscorlib]System.Collections.IEnumerator enumerator,
[3] class [mscorlib]System.IDisposable disposer)
// if(string.IsNullOrEmpty(strValue)) return defaultValue;
ldarg strValue
call bool [mscorlib]System.String::IsNullOrEmpty(string)
brfalse.s HASVALUE
br RETURNDEF // return default it empty
// foreach (T item in Enum.GetValues(typeof(T)))
HASVALUE:
// Enum.GetValues.GetEnumerator()
ldtoken !!T
call class [mscorlib]System.Type [mscorlib]System.Type::GetTypeFromHandle(valuetype [mscorlib]System.RuntimeTypeHandle)
call class [mscorlib]System.Array [mscorlib]System.Enum::GetValues(class [mscorlib]System.Type)
callvirt instance class [mscorlib]System.Collections.IEnumerator [mscorlib]System.Array::GetEnumerator()
stloc enumerator
.try
{
CONDITION:
ldloc enumerator
callvirt instance bool [mscorlib]System.Collections.IEnumerator::MoveNext()
brfalse.s LEAVE
STATEMENTS:
// T item = (T)Enumerator.Current
ldloc enumerator
callvirt instance object [mscorlib]System.Collections.IEnumerator::get_Current()
unbox.any !!T
stloc temp
ldloca.s temp
constrained. !!T
// if (item.ToString().ToLower().Equals(value.Trim().ToLower())) return item;
callvirt instance string [mscorlib]System.Object::ToString()
callvirt instance string [mscorlib]System.String::ToLower()
ldarg strValue
callvirt instance string [mscorlib]System.String::Trim()
callvirt instance string [mscorlib]System.String::ToLower()
callvirt instance bool [mscorlib]System.String::Equals(string)
brfalse.s CONDITION
ldloc temp
stloc return_value
leave.s RETURNVAL
LEAVE:
leave.s RETURNDEF
}
finally
{
// ArrayList's Enumerator may or may not inherit from IDisposable
ldloc enumerator
isinst [mscorlib]System.IDisposable
stloc.s disposer
ldloc.s disposer
ldnull
ceq
brtrue.s LEAVEFINALLY
ldloc.s disposer
callvirt instance void [mscorlib]System.IDisposable::Dispose()
LEAVEFINALLY:
endfinally
}
RETURNDEF:
ldarg defaultValue
stloc return_value
RETURNVAL:
ldloc return_value
ret
}
}
如果它是有效的C#,它将生成如下所示的函数:
T GetEnumFromString<T>(string valueString, T defaultValue) where T : Enum
然后使用以下C#代码:
using MyThing;
// stuff...
private enum MyEnum { Yes, No, Okay }
static void Main(string[] args)
{
Thing.GetEnumFromString("No", MyEnum.Yes); // returns MyEnum.No
Thing.GetEnumFromString("Invalid", MyEnum.Okay); // returns MyEnum.Okay
Thing.GetEnumFromString("AnotherInvalid", 0); // compiler error, not an Enum
}
不幸的是,这意味着用MSIL而不是C#来编写代码的这一部分,唯一的附加好处是您可以通过约束此方法System.Enum
。这也很让人讨厌,因为它被编译成一个单独的程序集。但是,这并不意味着您必须采用这种方式进行部署。
通过删除行.assembly MyThing{}
并按以下方式调用ilasm:
ilasm.exe /DLL /OUTPUT=MyThing.netmodule
您将得到一个netmodule而不是一个程序集。
不幸的是,VS2010(显然是更早的版本)不支持添加netmodule引用,这意味着在调试时必须将其保留在2个单独的程序集中。可以将它们添加为程序集一部分的唯一方法是自己使用/addmodule:{files}
命令行参数运行csc.exe 。在MSBuild脚本中不会太痛苦。当然,如果您勇敢或愚蠢,则可以每次手动手动运行csc。随着多个程序集需要访问它,它肯定会变得更加复杂。
因此,可以在.Net中完成。值得付出额外的努力吗?嗯,我想我会让你决定那个。
F#解决方案作为替代
额外的功劳:事实证明,enum
除了MSIL:F#之外,至少还可以使用一种其他.NET语言进行通用限制。
type MyThing =
static member GetEnumFromString<'T when 'T :> Enum> str defaultValue: 'T =
/// protect for null (only required in interop with C#)
let str = if isNull str then String.Empty else str
Enum.GetValues(typedefof<'T>)
|> Seq.cast<_>
|> Seq.tryFind(fun v -> String.Compare(v.ToString(), str.Trim(), true) = 0)
|> function Some x -> x | None -> defaultValue
由于这是一种众所周知的语言,具有完整的Visual Studio IDE支持,因此更易于维护,但是您在解决方案中仍然需要一个单独的项目。但是,它自然会产生非常不同的IL(代码是非常不同的),并且它依赖于FSharp.Core
库,就像其他任何外部库一样,它也需要成为您发行版的一部分。
使用它的方式(与MSIL解决方案基本相同),并证明它在其他同义结构上正确失败:
// works, result is inferred to have type StringComparison
var result = MyThing.GetEnumFromString("OrdinalIgnoreCase", StringComparison.Ordinal);
// type restriction is recognized by C#, this fails at compile time
var result = MyThing.GetEnumFromString("OrdinalIgnoreCase", 42);