Convert.ChangeType и преобразование в перечисления?
Я получил значение Int16
из базы данных и должен преобразовать его в тип перечисления. Это, к сожалению, сделано в слое кода, который очень мало знает об объектах, за исключением того, что он может собрать через отражение.
Таким образом, он заканчивает вызов Convert.ChangeType
, который выходит из строя с недопустимым исключением исключения.
Я нашел то, что считаю вонючим обходным путем, например:
String name = Enum.GetName(destinationType, value);
Object enumValue = Enum.Parse(destinationType, name, false);
Есть ли лучший способ, так что мне не нужно перемещаться по этой операции String?
Здесь короткая, но полная программа, которую можно использовать, если кому-то нужно поэкспериментировать:
using System;
public class MyClass
{
public enum DummyEnum
{
Value0,
Value1
}
public static void Main()
{
Int16 value = 1;
Type destinationType = typeof(DummyEnum);
String name = Enum.GetName(destinationType, value);
Object enumValue = Enum.Parse(destinationType, name, false);
Console.WriteLine("" + value + " = " + enumValue);
}
}
Ответы
Ответ 1
Enum.ToObject(....
- это то, что вы ищете!
С#
StringComparison enumValue = (StringComparison)Enum.ToObject(typeof(StringComparison), 5);
VB.NET
Dim enumValue As StringComparison = CType([Enum].ToObject(GetType(StringComparison), 5), StringComparison)
Если вы сделаете много конверсий Enum, попробуйте использовать следующий класс, он сохранит вам много кода.
public class Enum<EnumType> where EnumType : struct, IConvertible
{
/// <summary>
/// Retrieves an array of the values of the constants in a specified enumeration.
/// </summary>
/// <returns></returns>
/// <remarks></remarks>
public static EnumType[] GetValues()
{
return (EnumType[])Enum.GetValues(typeof(EnumType));
}
/// <summary>
/// Converts the string representation of the name or numeric value of one or more enumerated constants to an equivalent enumerated object.
/// </summary>
/// <param name="name"></param>
/// <returns></returns>
/// <remarks></remarks>
public static EnumType Parse(string name)
{
return (EnumType)Enum.Parse(typeof(EnumType), name);
}
/// <summary>
/// Converts the string representation of the name or numeric value of one or more enumerated constants to an equivalent enumerated object.
/// </summary>
/// <param name="name"></param>
/// <param name="ignoreCase"></param>
/// <returns></returns>
/// <remarks></remarks>
public static EnumType Parse(string name, bool ignoreCase)
{
return (EnumType)Enum.Parse(typeof(EnumType), name, ignoreCase);
}
/// <summary>
/// Converts the specified object with an integer value to an enumeration member.
/// </summary>
/// <param name="value"></param>
/// <returns></returns>
/// <remarks></remarks>
public static EnumType ToObject(object value)
{
return (EnumType)Enum.ToObject(typeof(EnumType), value);
}
}
Теперь вместо написания (StringComparison)Enum.ToObject(typeof(StringComparison), 5);
вы можете просто написать Enum<StringComparison>.ToObject(5);
.
Ответ 2
На основе ответа @Peter здесь приведен метод преобразования Nullable<int>
в Enum
:
public static class EnumUtils
{
public static bool TryParse<TEnum>(int? value, out TEnum result)
where TEnum: struct, IConvertible
{
if(!value.HasValue || !Enum.IsDefined(typeof(TEnum), value)){
result = default(TEnum);
return false;
}
result = (TEnum)Enum.ToObject(typeof(TEnum), value);
return true;
}
}
Использование EnumUtils.TryParse<YourEnumType>(someNumber, out result)
становится полезным для многих сценариев. Например, WebApi Controller в Asp.NET не имеет защиты по умолчанию от неверных параметров Enum. Asp.NET будет использовать значение default(YourEnumType)
, даже если некоторые пропускают null
, -1000
, 500000
, "garbage string"
или полностью игнорируют параметр. Более того, ModelState
будет действовать во всех этих случаях, поэтому одним из решений является использование типа int?
с пользовательской проверкой
public class MyApiController: Controller
{
[HttpGet]
public IActionResult Get(int? myEnumParam){
MyEnumType myEnumParamParsed;
if(!EnumUtils.TryParse<MyEnumType>(myEnumParam, out myEnumParamParsed)){
return BadRequest($"Error: parameter '{nameof(myEnumParam)}' is not specified or incorrect");
}
return this.Get(washingServiceTypeParsed);
}
private IActionResult Get(MyEnumType myEnumParam){
// here we can guarantee that myEnumParam is valid
}
Ответ 3
Если вы сохраняете Enum в DataTable, но не знаете, какой столбец является перечислением и который является строкой /int, вы можете получить доступ к этому значению следующим образом:
foreach (DataRow dataRow in myDataTable.Rows)
{
Trace.WriteLine("=-=-=-=-=-=-=-=-=-=-=-=-=-=-=");
foreach (DataColumn dataCol in myDataTable.Columns)
{
object v = dataRow[dataCol];
Type t = dataCol.DataType;
bool e = false;
if (t.IsEnum) e = true;
Trace.WriteLine((dataCol.ColumnName + ":").PadRight(30) +
(e ? Enum.ToObject(t, v) : v));
}
}