1. 程式人生 > >[小技巧]C#中如何為枚舉類型添加描述方法

[小技巧]C#中如何為枚舉類型添加描述方法

tomat 分享圖片 2個 stat 第一個 不同的 esc his !=

技術分享圖片

背景

在我們的日常開發中,我們會經常使用枚舉類型。有時我們只需要顯示枚舉的值或者枚舉值對應名稱, 但是在某些場景下,我們可能需要將枚舉值顯示為不同的字符串。

例: 當前我們有如下枚舉Level

    public enum Level
    {
        //Bad
        B = -1,

        //Normal
        N = 0,

        //Good
        G = 1,

        //Very Good 
        VG = 2
    }

這個枚舉有4個可選值B, N, G, VG。 現在我們希望用Bad, Normal, Good, Very Good作為B, N, G, VG的顯示值。

那我們會怎麽做呢?通常我們最常想到的就是針對Level枚舉類型編寫一個擴展方法。

    public static class LevelEnumExtension
    {
        public static string ToDescription(this Level level)
        {
            switch (level)
            {
                case Level.B:
                    return "Bad";
                case Level.G:
                    return "Good";
                case Level.N:
                    return "Normal";
                case Level.VG:
                    return "Very Good";
                default:
                    return "Normal";
            }
        }
    }

以上的代碼在我們的項目中很常用。但是這裏有2個潛在的問題:

  • 我們的項目中可能不止一種枚舉類型,所以我們可能就需要為每一種類型都添加一個對應的擴展方法。
  • 枚舉值和枚舉的顯示值的代碼位置是分離的,如果你查找枚舉值對應的顯示值,你就要先去找到對應的枚舉擴展方法。

那麽如何改進這部分代碼,從而消除上述2個問題呢,這時候我們就要引入.NET中的文本描述屬性類DescriptionAttribute

使用DescriptionAttribute重構代碼

其實.NET中已經提供了一個文本描述屬性類DescriptionAttribute, 這個屬性類的構造函數可以接收一段文字描述。

下面我們使用DescriptionAttribute

來改造Level枚舉類型。

    public enum Level
    {
        //Bad
        [Description("Bad")]
        B = -1,

        //Normal
        [Description("Normal")]
        N = 0,

        //Good
        [Description("Good")]
        G = 1,

        //Very Good 
        [Description("Very Good")]
        VG = 2
    }

這樣我們上面提到的第二個問題就解決了,現在Level枚舉類型的枚舉值和顯示值就都封裝在了一起。

那麽第一個問題該怎麽解決呢?

這裏我們可以針對Enum類型添加擴展方法,並使用反射讀取當前枚舉值所對應的顯示值

    public static class EnumExtension
    {
        public static string ToDescription(this Enum val)
        {
            var type = val.GetType();

            var memberInfo = type.GetMember(val.ToString());
            
            var attributes = memberInfo[0].GetCustomAttributes(typeof(DescriptionAttribute), false);

            if (attributes == null || attributes.Length != 1)
            {
                //如果沒有定義描述,就把當前枚舉值的對應名稱返回
                return val.ToString();
            }

            return (attributes.Single() as DescriptionAttribute).Description;
        }
    }

由於Enum類型是所有枚舉類型的基類型,所以所有的枚舉類型都可以使用這個擴展方法。

總結

本篇博文中,我們講解了如果如何.NET內置的文本描述屬性類DescriptionAttribute來生成枚舉值對應的文本,它不僅可以減少重復代碼,還可以讓整個枚舉類型的內聚性更高。

[小技巧]C#中如何為枚舉類型添加描述方法