获取枚举值的属性


483

我想知道是否有可能获取enum值的属性而不是其enum本身?例如,假设我有以下内容enum

using System.ComponentModel; // for DescriptionAttribute

enum FunkyAttributesEnum
{
    [Description("Name With Spaces1")]
    NameWithoutSpaces1,    
    [Description("Name With Spaces2")]
    NameWithoutSpaces2
}

我想要的是给定的枚举类型,产生2个元组的枚举字符串值及其描述。

价值很容易:

Array values = System.Enum.GetValues(typeof(FunkyAttributesEnum));
foreach (int value in values)
    Tuple.Value = Enum.GetName(typeof(FunkyAttributesEnum), value);

但是,如何获取描述属性的值进行填充Tuple.Desc?我可以考虑如果Attribute属于enum自身,那么该怎么做,但是我对于如何从的值中获取它感到困惑enum




2
Description所需的名称空间是System.ComponentModel
John M

您也可以只不使用System.ComponentModel,而只使用自己的属性类型。真的没有什么特别的DescriptionAttribute
jrh '18 -10-8

Answers:


482

这应该做您需要的。

var enumType = typeof(FunkyAttributesEnum);
var memberInfos = enumType.GetMember(FunkyAttributesEnum.NameWithoutSpaces1.ToString());
var enumValueMemberInfo = memberInfos.FirstOrDefault(m => m.DeclaringType == enumType);
var valueAttributes = 
      enumValueMemberInfo.GetCustomAttributes(typeof(DescriptionAttribute), false);
var description = ((DescriptionAttribute)valueAttributes[0]).Description;

10
(可选)使用type.GetFields(BindingFlags.Public | BindingFlags.Static)一次获取所有memInfos。
TrueWill 2011年

4
我不得不去typeof(FunkyAttributesEnum),但除此之外,它工作得很好。谢谢。
格雷格·兰德尔

@AlexK我看不到Enum类具有NameWithoutSpaces1属性。FunkyAttributesEnum.NameWithoutSpaces1来自何处?

2
@Don,这是OP的问题中的枚举成员名称。
MEMark 2014年

287

这段代码应该为您提供任何不错的扩展方法,让您可以检索任何通用属性的枚举。我相信它与上面的lambda函数有所不同,因为它使用起来更简单,并且稍微有点-您只需要传递泛型类型即可。

public static class EnumHelper
{
    /// <summary>
    /// Gets an attribute on an enum field value
    /// </summary>
    /// <typeparam name="T">The type of the attribute you want to retrieve</typeparam>
    /// <param name="enumVal">The enum value</param>
    /// <returns>The attribute of type T that exists on the enum value</returns>
    /// <example><![CDATA[string desc = myEnumVariable.GetAttributeOfType<DescriptionAttribute>().Description;]]></example>
    public static T GetAttributeOfType<T>(this Enum enumVal) where T:System.Attribute
    {
        var type = enumVal.GetType();
        var memInfo = type.GetMember(enumVal.ToString());
        var attributes = memInfo[0].GetCustomAttributes(typeof(T), false);
        return (attributes.Length > 0) ? (T)attributes[0] : null;
    }
}

19
用法如下:字符串desc = myEnumVariable.GetAttributeOfType <DescriptionAttribute>()。Description;
Brad Rem 2013年

2
我比Scott更喜欢这个,因为这里的用法比较干净(无需键入内容),所以+1 :)
nawfal

3
如果没有属性,这不会抛出IndexOutOfRangeException吗?
Erik Philips

6
最好对memInfo使用type.GetMember(Enum.GetName(type,enumVal)),因为enumVal.ToString()对于不同的语言环境可能并不可靠。
林松杨

2
调用GetCustomAttributes()然后获取第一个元素而不是调用有GetCustomAttribute()什么意义?
tigrou

81

这是使用lambda进行选择的通用实现

public static Expected GetAttributeValue<T, Expected>(this Enum enumeration, Func<T, Expected> expression)
    where T : Attribute
{
    T attribute =
      enumeration
        .GetType()
        .GetMember(enumeration.ToString())
        .Where(member => member.MemberType == MemberTypes.Field)
        .FirstOrDefault()
        .GetCustomAttributes(typeof(T), false)
        .Cast<T>()
        .SingleOrDefault();

    if (attribute == null)
        return default(Expected);

    return expression(attribute);
}

这样称呼它:

string description = targetLevel.GetAttributeValue<DescriptionAttribute, string>(x => x.Description);

4
这很棒。如果给定的枚举值是一个组合(由允许FlagsAttribute),我们只需要小心。在这种情况下,enumeration.GetType().GetMember(enumeration.ToString())[0]将失败。
remio 2011年

您可以写出最短的代码:value.GetType().GetField(value.ToString()).GetCustomAttributes(false).OfType<T>‌​().SingleOrDefault(),但是必须承认您的显式方式更好。
nawfal 2013年

2
我还添加了公共静态字符串GetDescription(此Enum枚举){return enumeration.GetAttributeValue <DescriptionAttribute,String>(x => x.Description); }这样,它的targetLevel.GetDescription();
MarkKGreenway 2014年

65

我在这里合并了几个答案,以创建一个更具扩展性的解决方案。我提供它是为了以防将来对其他人有帮助。原始张贴在这里

using System;
using System.ComponentModel;

public static class EnumExtensions {

    // This extension method is broken out so you can use a similar pattern with 
    // other MetaData elements in the future. This is your base method for each.
    public static T GetAttribute<T>(this Enum value) where T : Attribute {
        var type = value.GetType();
        var memberInfo = type.GetMember(value.ToString());
        var attributes = memberInfo[0].GetCustomAttributes(typeof(T), false);
        return attributes.Length > 0 
          ? (T)attributes[0]
          : null;
    }

    // This method creates a specific call to the above method, requesting the
    // Description MetaData attribute.
    public static string ToName(this Enum value) {
        var attribute = value.GetAttribute<DescriptionAttribute>();
        return attribute == null ? value.ToString() : attribute.Description;
    }

}

此解决方案在Enum上创建一对扩展方法。第一个允许您使用反射来检索与您的值关联的任何属性。第二个专门调用检索DescriptionAttribute并返回其Description值。

例如,考虑使用DescriptionAttribute来自System.ComponentModel

using System.ComponentModel;

public enum Days {
    [Description("Sunday")]
    Sun,
    [Description("Monday")]
    Mon,
    [Description("Tuesday")]
    Tue,
    [Description("Wednesday")]
    Wed,
    [Description("Thursday")]
    Thu,
    [Description("Friday")]
    Fri,
    [Description("Saturday")]
    Sat
}

要使用上述扩展方法,您现在只需调用以下命令:

Console.WriteLine(Days.Mon.ToName());

要么

var day = Days.Mon;
Console.WriteLine(day.ToName());

在最后一行,您的意思是“ attribute.Description”?返回属性== null?value.ToString():attribute.Description;
Jeson Martajaya 2014年

2
我喜欢这个解决方案,但是其中有一个错误。GetAttribute方法假定枚举值具有Description属性,因此当属性长度为0时引发异常。替换“ return(T)attributes [0];”。与“返回(attributes.Length> 0?(T)attributes [0]:null);”
西蒙·吉默

@SimonGymer感谢您的建议-我已经进行了相应的更新。:)
Troy Alford '18

38

除了AdamCrawford响应之外,我还创建了一种更专业的扩展方法,将其提要以获取描述。

public static string GetAttributeDescription(this Enum enumValue)
{
    var attribute = enumValue.GetAttributeOfType<DescriptionAttribute>();
    return attribute == null ? String.Empty : attribute.Description;
} 

因此,为了获得描述,您可以使用原始扩展方法作为

string desc = myEnumVariable.GetAttributeOfType<DescriptionAttribute>().Description

或者您可以简单地将扩展方法调用为:

string desc = myEnumVariable.GetAttributeDescription();

希望可以使您的代码更具可读性。


16

一口流利的班轮...

在这里,我使用DisplayAttribute包含NameDescription属性的。

public static DisplayAttribute GetDisplayAttributesFrom(this Enum enumValue, Type enumType)
{
    return enumType.GetMember(enumValue.ToString())
                   .First()
                   .GetCustomAttribute<DisplayAttribute>();
}

public enum ModesOfTransport
{
    [Display(Name = "Driving",    Description = "Driving a car")]        Land,
    [Display(Name = "Flying",     Description = "Flying on a plane")]    Air,
    [Display(Name = "Sea cruise", Description = "Cruising on a dinghy")] Sea
}

void Main()
{
    ModesOfTransport TransportMode = ModesOfTransport.Sea;
    DisplayAttribute metadata = TransportMode.GetDisplayAttributesFrom(typeof(ModesOfTransport));
    Console.WriteLine("Name: {0} \nDescription: {1}", metadata.Name, metadata.Description);
}

输出量

Name: Sea cruise 
Description: Cruising on a dinghy

2
我也用这个,它是所有答案中最干净的!+1
马菲

这似乎很有用!Thnx
Irf

7

这是从Display属性获取信息的代码。它使用通用方法检索属性。如果找不到该属性,它将枚举值转换为字符串,并且pascal / camel大小写转换为标题大小写(在此处获得代码)

public static class EnumHelper
{
    // Get the Name value of the Display attribute if the   
    // enum has one, otherwise use the value converted to title case.  
    public static string GetDisplayName<TEnum>(this TEnum value)
        where TEnum : struct, IConvertible
    {
        var attr = value.GetAttributeOfType<TEnum, DisplayAttribute>();
        return attr == null ? value.ToString().ToSpacedTitleCase() : attr.Name;
    }

    // Get the ShortName value of the Display attribute if the   
    // enum has one, otherwise use the value converted to title case.  
    public static string GetDisplayShortName<TEnum>(this TEnum value)
        where TEnum : struct, IConvertible
    {
        var attr = value.GetAttributeOfType<TEnum, DisplayAttribute>();
        return attr == null ? value.ToString().ToSpacedTitleCase() : attr.ShortName;
    }

    /// <summary>
    /// Gets an attribute on an enum field value
    /// </summary>
    /// <typeparam name="TEnum">The enum type</typeparam>
    /// <typeparam name="T">The type of the attribute you want to retrieve</typeparam>
    /// <param name="value">The enum value</param>
    /// <returns>The attribute of type T that exists on the enum value</returns>
    private static T GetAttributeOfType<TEnum, T>(this TEnum value)
        where TEnum : struct, IConvertible
        where T : Attribute
    {

        return value.GetType()
                    .GetMember(value.ToString())
                    .First()
                    .GetCustomAttributes(false)
                    .OfType<T>()
                    .LastOrDefault();
    }
}

这是用于转换为标题大小写的字符串的扩展方法:

    /// <summary>
    /// Converts camel case or pascal case to separate words with title case
    /// </summary>
    /// <param name="s"></param>
    /// <returns></returns>
    public static string ToSpacedTitleCase(this string s)
    {
        //https://stackoverflow.com/a/155486/150342
        CultureInfo cultureInfo = Thread.CurrentThread.CurrentCulture;
        TextInfo textInfo = cultureInfo.TextInfo;
        return textInfo
           .ToTitleCase(Regex.Replace(s, 
                        "([a-z](?=[A-Z0-9])|[A-Z](?=[A-Z][a-z]))", "$1 "));
    }

4

我实现了此扩展方法,以从枚举值获取描述。它适用于所有枚举。

public static class EnumExtension
{
    public static string ToDescription(this System.Enum value)
    {
        FieldInfo fi = value.GetType().GetField(value.ToString());
        var attributes = (DescriptionAttribute[])fi.GetCustomAttributes(typeof(DescriptionAttribute), false);
        return attributes.Length > 0 ? attributes[0].Description : value.ToString();
    }
}

相同解决方案的通用版本已经发布。imo,更好。
nawfal

4

从枚举中获取字典。

public static IDictionary<string, int> ToDictionary(this Type enumType)
{
    return Enum.GetValues(enumType)
    .Cast<object>()
    .ToDictionary(v => ((Enum)v).ToEnumDescription(), k => (int)k); 
}

现在这样称呼...

var dic = typeof(ActivityType).ToDictionary();

EnumDecription Ext方法

public static string ToEnumDescription(this Enum en) //ext method
{
    Type type = en.GetType();
    MemberInfo[] memInfo = type.GetMember(en.ToString());
    if (memInfo != null && memInfo.Length > 0)
    {
        object[] attrs = memInfo[0].GetCustomAttributes(typeof(DescriptionAttribute), false);
        if (attrs != null && attrs.Length > 0)
            return ((DescriptionAttribute)attrs[0]).Description;
    }
    return en.ToString();
}

public enum ActivityType
{
    [Description("Drip Plan Email")]
    DripPlanEmail = 1,
    [Description("Modification")]
    Modification = 2,
    [Description("View")]
    View = 3,
    [Description("E-Alert Sent")]
    EAlertSent = 4,
    [Description("E-Alert View")]
    EAlertView = 5
}

3

这是AdamCrawford答案的.NET Core版本,使用System.Reflection.TypeExtensions ;

public static class EnumHelper
{
    /// <summary>
    /// Gets an attribute on an enum field value
    /// </summary>
    /// <typeparam name="T">The type of the attribute you want to retrieve</typeparam>
    /// <param name="enumVal">The enum value</param>
    /// <returns>The attribute of type T that exists on the enum value</returns>
    /// <example>string desc = myEnumVariable.GetAttributeOfType<DescriptionAttribute>().Description;</example>
    public static T GetAttributeOfType<T>(this Enum enumVal) where T : System.Attribute
    {
        var type = enumVal.GetType();
        var memInfo = type.GetMember(enumVal.ToString());
        IEnumerable<Attribute> attributes = memInfo[0].GetCustomAttributes(typeof(T), false);
        return (T)attributes?.ToArray()[0];
    }
}

我不相信.NET Core(或更确切地说,现在是Standard)具有GetMember,所以我不确定这将如何工作。
杰夫

它在System.Reflection.TypeExtensions中,我已经修改了答案以列出它。
威尼娅

1
知道了,谢谢。我以为可能会有一些扩展。
杰夫

3

添加了针对Net Framework和NetCore的解决方案。

我将此用于我的Net Framework实现:

public static class EnumerationExtension
{
    public static string Description( this Enum value )
    {
        // get attributes  
        var field = value.GetType().GetField( value.ToString() );
        var attributes = field.GetCustomAttributes( typeof( DescriptionAttribute ), false );

        // return description
        return attributes.Any() ? ( (DescriptionAttribute)attributes.ElementAt( 0 ) ).Description : "Description Not Found";
    }
}

这不适用于NetCore,因此我对其进行了修改以做到这一点:

public static class EnumerationExtension
{
    public static string Description( this Enum value )
    {
        // get attributes  
        var field = value.GetType().GetField( value.ToString() );
        var attributes = field.GetCustomAttributes( false );

        // Description is in a hidden Attribute class called DisplayAttribute
        // Not to be confused with DisplayNameAttribute
        dynamic displayAttribute = null;

        if (attributes.Any())
        {
            displayAttribute = attributes.ElementAt( 0 );
        }

        // return description
        return displayAttribute?.Description ?? "Description Not Found";
    }
}

枚举示例:

public enum ExportTypes
{
    [Display( Name = "csv", Description = "text/csv" )]
    CSV = 0
}

静态添加的示例用法:

var myDescription = myEnum.Description();

2

利用某些较新的C#语言功能,可以减少行数:

public static TAttribute GetEnumAttribute<TAttribute>(this Enum enumVal) where TAttribute : Attribute
{
    var memberInfo = enumVal.GetType().GetMember(enumVal.ToString());
    return memberInfo[0].GetCustomAttributes(typeof(TAttribute), false).OfType<TAttribute>().FirstOrDefault();
}

public static string GetEnumDescription(this Enum enumValue) => enumValue.GetEnumAttribute<DescriptionAttribute>()?.Description ?? enumValue.ToString();

2

我这个答案是从一个很棒的枚举属性设置一个组合框。

然后,我需要对相反的代码进行编码,以便可以从框中获得选择并以正确的类型返回枚举。

我还修改了代码以处理缺少属性的情况

为了下一个人的利益,这是我的最终解决方案

public static class Program
{
   static void Main(string[] args)
    {
       // display the description attribute from the enum
       foreach (Colour type in (Colour[])Enum.GetValues(typeof(Colour)))
       {
            Console.WriteLine(EnumExtensions.ToName(type));
       }

       // Get the array from the description
       string xStr = "Yellow";
       Colour thisColour = EnumExtensions.FromName<Colour>(xStr);

       Console.ReadLine();
    }

   public enum Colour
   {
       [Description("Colour Red")]
       Red = 0,

       [Description("Colour Green")]
       Green = 1,

       [Description("Colour Blue")]
       Blue = 2,

       Yellow = 3
   }
}

public static class EnumExtensions
{

    // This extension method is broken out so you can use a similar pattern with 
    // other MetaData elements in the future. This is your base method for each.
    public static T GetAttribute<T>(this Enum value) where T : Attribute
    {
        var type = value.GetType();
        var memberInfo = type.GetMember(value.ToString());
        var attributes = memberInfo[0].GetCustomAttributes(typeof(T), false);

        // check if no attributes have been specified.
        if (((Array)attributes).Length > 0)
        {
            return (T)attributes[0];
        }
        else
        {
            return null;
        }
    }

    // This method creates a specific call to the above method, requesting the
    // Description MetaData attribute.
    public static string ToName(this Enum value)
    {
        var attribute = value.GetAttribute<DescriptionAttribute>();
        return attribute == null ? value.ToString() : attribute.Description;
    }

    /// <summary>
    /// Find the enum from the description attribute.
    /// </summary>
    /// <typeparam name="T"></typeparam>
    /// <param name="desc"></param>
    /// <returns></returns>
    public static T FromName<T>(this string desc) where T : struct
    {
        string attr;
        Boolean found = false;
        T result = (T)Enum.GetValues(typeof(T)).GetValue(0);

        foreach (object enumVal in Enum.GetValues(typeof(T)))
        {
            attr = ((Enum)enumVal).ToName();

            if (attr == desc)
            {
                result = (T)enumVal;
                found = true;
                break;
            }
        }

        if (!found)
        {
            throw new Exception();
        }

        return result;
    }
}

}


1
伙计,我见过这么多愚蠢且无法解释的解决方案,而您的解决方案使它丧命。非常感谢<3
Kadaj '16

2

如果您enum的值包含类似的值Equals,则可能会在此处的许多答案中使用某些扩展名遇到一些错误。这是因为通常假设该方法typeof(YourEnum).GetMember(YourEnum.Value)只会返回一个值,即MemberInfo您的的enum。这是亚当·克劳福德的答案稍微更安全的版本。

public static class AttributeExtensions
{
    #region Methods

    public static T GetAttribute<T>(this Enum enumValue) where T : Attribute
    {
        var type = enumValue.GetType();
        var memberInfo = type.GetMember(enumValue.ToString());
        var member = memberInfo.FirstOrDefault(m => m.DeclaringType == type);
        var attribute = Attribute.GetCustomAttribute(member, typeof(T), false);
        return attribute is T ? (T)attribute : null;
    }

    #endregion
}

1

此扩展方法将使用其XmlEnumAttribute获得枚举值的字符串表示形式。如果不存在XmlEnumAttribute,它将回退到enum.ToString()。

public static string ToStringUsingXmlEnumAttribute<T>(this T enumValue)
    where T: struct, IConvertible
{
    if (!typeof(T).IsEnum)
    {
        throw new ArgumentException("T must be an enumerated type");
    }

    string name;

    var type = typeof(T);

    var memInfo = type.GetMember(enumValue.ToString());

    if (memInfo.Length == 1)
    {
        var attributes = memInfo[0].GetCustomAttributes(typeof(System.Xml.Serialization.XmlEnumAttribute), false);

        if (attributes.Length == 1)
        {
            name = ((System.Xml.Serialization.XmlEnumAttribute)attributes[0]).Name;
        }
        else
        {
            name = enumValue.ToString();
        }
    }
    else
    {
        name = enumValue.ToString();
    }

    return name;
}

1

如果您需要完整的名称列表,可以执行以下操作

typeof (PharmacyConfigurationKeys).GetFields()
        .Where(x => x.GetCustomAttributes(false).Any(y => typeof(DescriptionAttribute) == y.GetType()))
        .Select(x => ((DescriptionAttribute)x.GetCustomAttributes(false)[0]).Description);

0

各位,如果有帮助,我将与您分享我的解决方案:自定义属性的定义:

    [AttributeUsage(AttributeTargets.Field,AllowMultiple = false)]
public class EnumDisplayName : Attribute
{
    public string Name { get; private set; }
    public EnumDisplayName(string name)
    {
        Name = name;
    }
}

现在,因为我需要在HtmlHelper扩展的HtmlHelper定义内使用它:

public static class EnumHelper
{
    public static string EnumDisplayName(this HtmlHelper helper,EPriceType priceType)
    {
        //Get every fields from enum
        var fields = priceType.GetType().GetFields();
        //Foreach field skipping 1`st fieldw which keeps currently sellected value
        for (int i = 0; i < fields.Length;i++ )
        {
            //find field with same int value
            if ((int)fields[i].GetValue(priceType) == (int)priceType)
            {
                //get attributes of found field
                var attributes = fields[i].GetCustomAttributes(false);
                if (attributes.Length > 0)
                {
                    //return name of found attribute
                    var retAttr = (EnumDisplayName)attributes[0];
                    return retAttr.Name;
                }
            }
        }
        //throw Error if not found
        throw new Exception("Błąd podczas ustalania atrybutów dla typu ceny allegro");
    }
}

希望能帮助到你


0
    public enum DataFilters
    {
        [Display(Name= "Equals")]
        Equals = 1,// Display Name and Enum Name are same 
        [Display(Name= "Does Not Equal")]
        DoesNotEqual = 2, // Display Name and Enum Name are different             
    }

现在,在这种情况下它将产生错误1“等于”

public static string GetDisplayName(this Enum enumValue)
    {
        var enumMember = enumValue.GetType().GetMember(enumValue.ToString()).First();
        return enumMember.GetCustomAttribute<DisplayAttribute>() != null ? enumMember.GetCustomAttribute<DisplayAttribute>().Name : enumMember.Name;
    }

因此,如果它是相同的,则返回枚举名称而不是显示名称,因为如果显示名称和枚举名称相同,则enumMember.GetCustomAttribute()将为null。


0

或者,您可以执行以下操作:

List<SelectListItem> selectListItems = new List<SelectListItem>();

    foreach (var item in typeof(PaymentTerm).GetEnumValues())
    {
        var type = item.GetType();
        var name = type.GetField(item.ToString()).GetCustomAttributesData().FirstOrDefault()?.NamedArguments.FirstOrDefault().TypedValue.Value.ToString();
        selectListItems.Add(new SelectListItem(name, type.Name));

    }

0

这就是我无需在.NET core 3.1中使用自定义帮助程序或扩展程序即可解决的方法。

public enum YourEnum
{
    [Display(Name = "Suryoye means Arameans")]
    SURYOYE = 0,
    [Display(Name = "Oromoye means Syriacs")]
    OROMOYE = 1,
}

剃刀

@using Enumerations

foreach (var name in Html.GetEnumSelectList(typeof(YourEnum)))
{
    <h1>@name.Text</h1>
}

1
考虑使用比您如何解决“问题”更多的方法来回答问题-首先要确认问题并解释您认为如何解决“问题”。请记住,您的答案可能会在几年后脱离上下文,这几乎是无用的。在其中添加更多内容,添加一些上下文信息,可以使您的答案及其可能的历史/档案相关性
得以

0

绩效很重要

如果您想获得更好的性能,可以采用以下方法:

public static class AdvancedEnumExtensions
{
    /// <summary>
    /// Gets the custom attribute <typeparamref name="T"/> for the enum constant, if such a constant is defined and has such an attribute; otherwise null.
    /// </summary>
    public static T GetCustomAttribute<T>(this Enum value) where T : Attribute
    {
        return GetField(value)?.GetCustomAttribute<T>(inherit: false);
    }

    /// <summary>
    /// Gets the FieldInfo for the enum constant, if such a constant is defined; otherwise null.
    /// </summary>
    public static FieldInfo GetField(this Enum value)
    {
        ulong u64 = ToUInt64(value);
        return value
            .GetType()
            .GetFields(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Static)
            .Where(f => ToUInt64(f.GetRawConstantValue()) == u64)
            .FirstOrDefault();
    }

    /// <summary>
    /// Checks if an enum constant is defined for this enum value
    /// </summary>
    public static bool IsDefined(this Enum value)
    {
        return GetField(value) != null;
    }

    /// <summary>
    /// Converts the enum value to UInt64
    /// </summary>
    public static ulong ToUInt64(this Enum value) => ToUInt64((object)value);

    private static ulong ToUInt64(object value)
    {
        switch (Convert.GetTypeCode(value))
        {
            case TypeCode.SByte:
            case TypeCode.Int16:
            case TypeCode.Int32:
            case TypeCode.Int64:
                return unchecked((ulong)Convert.ToInt64(value, CultureInfo.InvariantCulture));

            case TypeCode.Byte:
            case TypeCode.UInt16:
            case TypeCode.UInt32:
            case TypeCode.UInt64:
            case TypeCode.Char:
            case TypeCode.Boolean:
                return Convert.ToUInt64(value, CultureInfo.InvariantCulture);

            default: throw new InvalidOperationException("UnknownEnumType");
        }
    }
}

为什么这会有更好的表现?

因为所有内置方法都使用与此非常相似的代码,除了它们还运行了许多我们不在乎的其他代码。通常,C#的Enum代码非常可怕。

上面的代码已经过Linq验证和简化,因此它仅包含我们关心的位。

为什么内置代码比较慢?

首先关于Enum.ToString()-vs- Enum.GetName(..)

始终使用后者。(或者更好,但两者都不行,如下所述。)

ToString()在内部使用后者,但同样,它还会执行其他一些我们不希望的事情,例如,尝试组合标志,打印数字等。我们只对枚举内部定义的常量感兴趣。

Enum.GetName依次获取所有字段,为所有名称创建一个字符串数组,在其所有RawConstantValues上使用上述ToUInt64创建所有值的UInt64数组,根据UInt64值对两个数组进行排序,最后从通过在UInt64数组中执行BinarySearch来查找所需值的索引来命名数组。

...然后我们将这些字段扔掉,然后将排序后的数组使用该名称再次查找该字段。

一句话:“ U!”


-1

或者,您可以执行以下操作:

Dictionary<FunkyAttributesEnum, string> description = new Dictionary<FunkyAttributesEnum, string>()
    {
      { FunkyAttributesEnum.NameWithoutSpaces1, "Name With Spaces1" },
      { FunkyAttributesEnum.NameWithoutSpaces2, "Name With Spaces2" },
    };

并获得以下描述:

string s = description[FunkyAttributesEnum.NameWithoutSpaces1];

我认为这是一种更有效的方式来完成您想完成的事情,因为不需要反思。


2
当然可以,但是反思并没有人们想象的那么糟糕。
Bryan Rowe

并不是说不好-我一直都在使用它。但是,它经常不必要地使用。:)
Ian P

44
该解决方案使描述远离枚举本身,从而至少造成两个大问题。首先,如果有人添加了一个新的枚举常量,他们将需要知道是否要去另一个地方在此处添加条目。属性是维护人员明确知道他们需要做什么的标志。我的第二个问题是它只是更多的代码。属性是紧凑的。
斯科比

1
@scott,但是它确实允许您指定自己的顺序,并排除您不想显示的值,这几乎总是我真正想要的值
Simon_Weaver

-2

您还可以定义一个枚举值,例如Name_Without_Spaces,并在需要描述时使用Name_Without_Spaces.ToString().Replace('_', ' ')下划线替换下划线。


8
这是一个非常微妙的解决方案。考虑使用@Bryan
Johann
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.