假设我们的使用场景是这样的:项目定义的枚举并不多,但是用其描述值很频繁,比如定义了一个用户性别枚举,用的地方很多,使用频率很高。
上面GetDescriptionByDictionaryWithLocak的方法中,第一句代码“if (_LockDictionary.ContainsKey(@this)) ”就是验证是否包含枚举值。在2.1的测试中执行了8000w次,其中只有80次(总共只有80个枚举值用于测试)需要这句代码“if (_LockDictionary.ContainsKey(@this)) ”,其余的直接取值就可了。基于这样的考虑,就有了上面的方法GetDescriptionByDictionaryWithException。
来测试一下,看看效果吧!
[Test] public void GetDescriptionByDictionaryWithException_Test() { var enums = this.GetTestEnums(); Console.WriteLine(enums.Count); TestHelper.InvokeAndWriteAll(() => { System.Threading.Tasks.Parallel.For(0, 1000000, (i, obj) => { foreach (var item in enums) { var a = item.GetDescriptionByDictionaryWithException(); } }); }); } //测试结果: 80 TimeSpan:1,208.0000ms MemoryUsed:230.9453KB CollectionCount(0):1.00
测试结果来看,基本上差不多,在时间上略微快乐一点点,1,208.0000ms:1,860.0000ms,执行8000w次快600毫秒,好像差别也不大啊,这是为什么呢?
这个其实就是Dictionary的问题了,Dictionary内部使用散列算法计算存储地址,其查找的时间复杂度为o(1),他的查找效果是非常快的,而本方法中利用了异常处理,异常捕获本身是有一定性能影响的。
2.3 推荐简单方案:ConcurrentDictionary
ConcurrentDictionary是一个线程安全的字典类,代码:
private static ConcurrentDictionary<Enum, string> _ConcurrentDictionary = new ConcurrentDictionary<Enum, string>(); public static string GetDescriptionByConcurrentDictionary(this Enum @this) { return _ConcurrentDictionary.GetOrAdd(@this, (key) => { var type = key.GetType(); var field = type.GetField(key.ToString()); return field == null ? key.ToString() : GetDescription(field); }); }
测试代码及测试结果:
[Test] public void GetDescriptionByConcurrentDictionary_Test() { var enums = this.GetTestEnums(); Console.WriteLine(enums.Count); TestHelper.InvokeAndWriteAll(() => { System.Threading.Tasks.Parallel.For(0, 1000000, (i, obj) => { foreach (var item in enums) { var a = item.GetDescriptionByConcurrentDictionary(); } }); }); } //测试结果: 80 TimeSpan:1,303.0000ms MemoryUsed:198.0859KB CollectionCount(0):1.00
2.4 正式的代码
综上所述,解决了性能问题、位域枚举问题的正式的代码:
/// <summary> /// 获取枚举的描述信息(Descripion)。 /// 支持位域,如果是位域组合值,多个按分隔符组合。 /// </summary> public static string GetDescription(this Enum @this) { return _ConcurrentDictionary.GetOrAdd(@this, (key) => { var type = key.GetType(); var field = type.GetField(key.ToString()); //如果field为null则应该是组合位域值, return field == null ? key.GetDescriptions() : GetDescription(field); }); } /// <summary> /// 获取位域枚举的描述,多个按分隔符组合 /// </summary> public static string GetDescriptions(this Enum @this, string separator = ",") { var names = @this.ToString().Split(','); string[] res = new string[names.Length]; var type = @this.GetType(); for (int i = 0; i < names.Length; i++) { var field = type.GetField(names[i].Trim()); if (field == null) continue; res[i] = GetDescription(field); } return string.Join(separator, res); } private static string GetDescription(FieldInfo field) { var att = System.Attribute.GetCustomAttribute(field, typeof(DescriptionAttribute), false); return att == null ? field.Name : ((DescriptionAttribute)att).Description; }
ps:.NET获取枚举值的描述
一、给枚举值定义描述的方式
public enum TimeOfDay { [Description("早晨")] Moning = 1, [Description("下午")] Afternoon = 2, [Description("晚上")] Evening = 3, }
二、获取枚举值的描述的方法
public static string GetDescriptionFromEnumValue(Type enumType, object enumValue) { try { object o = Enum.Parse(enumType, enumValue.ToString()); string name = o.ToString(); DescriptionAttribute[] customAttributes = (DescriptionAttribute[])enumType.GetField(name).GetCustomAttributes(typeof(DescriptionAttribute), false); if ((customAttributes != null) && (customAttributes.Length == 1)) { return customAttributes[0].Description; } return name; } catch { return "未知"; } }
三、获取枚举值的描述的方法的使用
string strMoning = GetDescriptionFromEnumValue( typeof (TimeOfDay) , 2 );
您可能感兴趣的文章: