Получить значение перечисления на основе значения имени XmlEnumAttribute
Мне нужна общая функция для извлечения имени или значения перечисления на основе свойства XmlEnumAttribute "Имя" перечисления. Например, у меня определено следующее перечисление:
Public Enum Currency
<XmlEnum("00")> CDN = 1
<XmlEnum("01")> USA= 2
<XmlEnum("02")> EUR= 3
<XmlEnum("03")> JPN= 4
End Enum
Первое значение перечисления валюты равно 1; имя перечисления - "CDN"; и значение свойства XMLEnumAttribute Name равно "00" .
Если у меня есть значение перечисления, я могу получить значение "Имя" XmlEnumAttribute, используя следующую общую функцию:
Public Function GetXmlAttrNameFromEnumValue(Of T)(ByVal pEnumVal As T) As String
Dim type As Type = pEnumVal.GetType
Dim info As FieldInfo = type.GetField([Enum].GetName(GetType(T), pEnumVal))
Dim att As XmlEnumAttribute = CType(info.GetCustomAttributes(GetType(XmlEnumAttribute), False)(0), XmlEnumAttribute) 'If there is an xmlattribute defined, return the name
Return att.Name
End Function
Таким образом, используя указанную выше функцию, я могу указать тип перечисления валюты, передать значение 1, а возвращаемое значение будет "00" .
Мне нужна функция для выполнения, если наоборот. Если у меня есть значение XmlEnumAttribute Name "00" , мне нужна функция для возврата перечисления валюты со значением 1. Точно так же полезной будет функция, которая возвращает имя переименования "CDN". Я мог бы просто проанализировать это, чтобы получить значение перечисления.
Любая помощь будет оценена по достоинству.
Ответы
Ответ 1
Требование решить эту ту же проблему привело меня к этому вопросу и ответу. Когда я развиваюсь в VB.NET, я переписал решение CkH в VB и изменил его, чтобы использовать вашу функцию GetXmlAttrNameFromEnumValue
.
Public Shared Function GetCode(Of T)(ByVal value As String) As T
For Each o As Object In System.Enum.GetValues(GetType(T))
Dim enumValue As T = CType(o, T)
If GetXmlAttrNameFromEnumValue(Of T)(enumValue).Equals(value, StringComparison.OrdinalIgnoreCase) Then
Return CType(o, T)
End If
Next
Throw New ArgumentException("No code exists for type " + GetType(T).ToString() + " corresponding to value of " + value)
End Function
Версия С#:
public static string GetXmlAttrNameFromEnumValue<T>(T pEnumVal)
{
// http://stackoverflow.com/q/3047125/194717
Type type = pEnumVal.GetType();
FieldInfo info = type.GetField(Enum.GetName(typeof(T), pEnumVal));
XmlEnumAttribute att = (XmlEnumAttribute)info.GetCustomAttributes(typeof(XmlEnumAttribute), false)[0];
//If there is an xmlattribute defined, return the name
return att.Name;
}
public static T GetCode<T>(string value)
{
// http://stackoverflow.com/a/3073272/194717
foreach (object o in System.Enum.GetValues(typeof(T)))
{
T enumValue = (T)o;
if (GetXmlAttrNameFromEnumValue<T>(enumValue).Equals(value, StringComparison.OrdinalIgnoreCase))
{
return (T)o;
}
}
throw new ArgumentException("No XmlEnumAttribute code exists for type " + typeof(T).ToString() + " corresponding to value of " + value);
}
Ответ 2
Я делаю что-то подобное с пользовательскими атрибутами, и я использую этот метод для получения EnumValue на основе значения атрибута. GetStringValue - мой собственный метод, аналогичный вашему примеру выше.
public static class Enums
{
public static T GetCode<T>(string value)
{
foreach (object o in System.Enum.GetValues(typeof(T)))
{
if (((Enum)o).GetStringValue().Equals(value, StringComparison.OrdinalIgnoreCase))
return (T)o;
}
throw new ArgumentException("No code exists for type " + typeof(T).ToString() + " corresponding to value of " + value);
}
}
В течение всего процесса я использую проверку этого сообщения и ответов: Расширение перечислений, Overkill?
Извините, что на С#, только что понял, что вы использовали VB.NET выше.
Ответ 3
Немного изменено:
http://www.wackylabs.net/2006/06/getting-the-xmlenumattribute-value-for-an-enum-field/
public static string ToString2 (this Enum e) {
// Get the Type of the enum
Type t = e.GetType ();
// Get the FieldInfo for the member field with the enums name
FieldInfo info = t.GetField (e.ToString ("G"));
// Check to see if the XmlEnumAttribute is defined on this field
if (!info.IsDefined (typeof (XmlEnumAttribute), false)) {
// If no XmlEnumAttribute then return the string version of the enum.
return e.ToString ("G");
}
// Get the XmlEnumAttribute
object[] o = info.GetCustomAttributes (typeof (XmlEnumAttribute), false);
XmlEnumAttribute att = (XmlEnumAttribute)o[0];
return att.Name;
}
Ответ 4
@Dean, @Jason и @Camron, спасибо за ваши решения. Ваши решения помогли мне решить мою проблему, где, учитывая имя XmlEnumAttribute, необходимо фактическое значение перечисления.
Мой вариант упоминается здесь.
Ответ 5
Здесь вариация, которая генерирует словарь из перечисления, позволяя вам потенциально кэшировать часть отражения, если вам нужно использовать его много.
/// <summary>
/// Generates a dictionary allowing you to get the csharp enum value
/// from the string value in the matching XmlEnumAttribute.
/// You need this to be able to dynamically set entries from a xsd:enumeration
/// when you've used xsd.exe to generate a .cs from the xsd.
/// https://msdn.microsoft.com/en-us/library/x6c1kb0s(v=vs.110).aspx
/// </summary>
/// <typeparam name="T">The xml enum type you want the mapping for</typeparam>
/// <returns>Mapping dictionary from attribute values (key) to the actual enum values</returns>
/// <exception cref="System.ArgumentException">T must be an enum</exception>
private static Dictionary<string, T> GetEnumMap<T>() where T : struct, IConvertible
{
if (!typeof(T).IsEnum)
{
throw new ArgumentException("T must be an enum");
}
var members = typeof(T).GetMembers();
var map = new Dictionary<string, T>();
foreach (var member in members)
{
var enumAttrib = member.GetCustomAttributes(typeof(XmlEnumAttribute), false).FirstOrDefault() as XmlEnumAttribute;
if (enumAttrib == null)
{
continue;
}
var xmlEnumValue = enumAttrib.Name;
var enumVal = ((FieldInfo)member).GetRawConstantValue();
map.Add(xmlEnumValue, (T)enumVal);
}
return map;
}
использование:
var map = GetEnumMap<Currency>();
return map["02"]; // returns Currency.EUR