[置顶] .NET下枚举类型的Save和Load分析

今天在写代码的时候,心血来潮对原来的字符串保存状态位的方式很不满意,对于代码里出现了 state == "1" 这样的状态判断很是不爽。那么理想中的判断是怎样的呢?很简单如你所想枚举类型。

public enum FormSate
{
   View,
   Modify
}
State == FormSate.View;



和"1"这样的硬代码比较起来,上面的代码看起来可读性很强。

.NET 枚举的应用分析

接下来,自然而然的会出现在ORM操作中,对于一个数据Model,我们需要与数据库打交道,那么它该怎么保存,又该保存为什么数据类型?首先该说说Enum对应的是什么基本类型?在.NET里,一个枚举类型默认是一个int,且默认是从0开始的,除非指定。(这里不探讨Enum的深入用法和语法分析)上面的代码实际上被定义为如下的格式。

public enum FormSate
{
   View = 0,
   Modify = 1
}

这样我们可以对一个枚举类进行基本的比较运算等等,比如"FormSate.View > FormSate.Mofiy"这样的比较。因此我们保存的时候就可以以int格式保存在数据库中。而读取的时候int是可以直接赋值给enum类型的,不过这里需要注意下的是这个赋值可不是用"="去赋值,而是反射SetValue。如果都用了"="号去赋值,我想肯定没用ORM,那么用一下的语句手动赋值。

Enum.Parse(typeof(FormSate), "1");

如果是用的ORM,大概都有提供对Enum的处理,看下API即可,不过原理大概都是反射,类似一下代码:

public class MyEnumClass
{
   public FormSate _a { get; set; }        
}
var b = new MyEnumClass();
var p = b.GetType().GetProperties().FirstOrDefault();
p.SetValue(b, 1, null);

枚举的自定义格式存储

在我的项目为例有这样的情况,所有状态位都是保存为varchar类型,会用"T"代表"True","F"代码"False"这样的存储,这时候需要改动上边的实现。首先,需要把数据库里的字段格式改为varchar,然后我们改动下枚举

public enum ResultState
{
   T = 0,//sucess
   F = 1//failed
}

这样的处理好意疑问没啥问题,可是代码的可读性变差了,从缩写上很难读懂。于是我们可以这样处理

public enum ResultState
{
   [Description("true")]
   T = 0,//sucess
   [Description("false")]
   F = 1//failed
}

给枚举值加上Description标签,用于解释属性的意思。这样做还是不能在引用这个枚举的时候见字就理解意思,只能找到定义查看。那么,或许咱们可以改造为从Description标签读取保存的信息?这样做的话,还可以充分的自定义,可以保存为任何我们想要的格式。

public enum ResultState
{
   [Description("T")]
   SUCCESS = 0,//sucess
   [Description("F")]
   FAILED = 1//failed
}


然后我们需要在读写ResultState的时候,进行反射。


public static class MyEnumExtensions
{
        public static string ToDescriptionString(this Enum val)
        {
            var attributes = (DescriptionAttribute[])val.GetType().GetField(val.ToString()).GetCustomAttributes(typeof(DescriptionAttribute), false);
            return attributes.Length > 0 ? attributes[0].Description : string.Empty;
        }
}


这里写了一个Enum类的扩展方法(什么是扩展方法?)ToDescriptionString对Enum扩展了取到Description标签内容的方法。利用了反射,对于反射的性能什么的,我只想说该用还是用,除非真的很看中那几毫秒的性能。


b._a.ToDescriptionString();


这样改造之后,对于数据库的读写部分,还是要进行改造,具体的思路和上面的常规思路一样,要么手动,要么在做ORM层的时候,对Enum进行特殊处理。不过这里我有一点点想法,自定义一个Attibute标签用于指示,如何处理Enum类型,这样对于每

种需求场景都可以自由的扩展和选择

public enum EnumOpration
{
    ByDescription,//利用Description标签
    Default,//默认的数字存储
    ByName//默认的存储属性名
}

public class MyEnumClass
{
   [EnumElement(opration = EnumOpration.ByName, DBType = DbType.String)]
   public FormSate _a { get; set; }        
}


这里opration指示了枚举取什么值保存,而DBType则指示保存的数据类型,当让要实现这一整套的机制,在ORM做Mapping的时候需要做很多工作,这里就不给出具体实现了。

枚举的另类替代方法

如何了解Java的肯定知道Java里是没有枚举的,那么该如何实现?.NET下这样的实现方式是否有可取之处呢?我转载了 Stackoverflow上的一个回答,建议大家观摩下,猛击这里


public class LogCategory
{
 private LogCategory(string value) { Value = value; }

 public string Value { get; set; }

 public static LogCategory Trace { get { return new LogCategory("Trace"); } }
 public static LogCategory Debug { get { return new LogCategory("Debug"); } }
 public static LogCategory Info { get { return new LogCategory("Info"); } }
 public static LogCategory Warning { get { return new LogCategory("Warning"); } }
 public static LogCategory Error { get { return new LogCategory("Error"); } }
}

public static void Write(string message, LogCategory logCategory)
{
   var log = new LogEntry { Message = message };
   Logger.Write(log, logCategory.Value);
}
Usage:

Logger.Write("This is almost like an enum.", LogCategory.Info);

不过这样的实现有个比较大的问题,数据存储的时候,需要做些工作不仅要反射出实体对象,还要用枚举属性的类型去反射创建枚举对象再赋值,不过也只是复杂了一步而已。

总结

以上便是今天我在看到枚举的时候想到的一些问题,可能大家会说都是废话,完全不需要想这么多。遇到枚举最简单的Save和Load方法还是直接保存int类型的值,这样省时省力。不过在遇到一些特殊的情况的时候,我想上面的自定义方式,也许对你有些帮助。如果以上有什么不对的地方,请指正。


我的独立博客地址:http://www.capqueen.name,欢迎交流。

 

posted @ 2013-08-23 18:56  pangbangb  阅读(168)  评论(0编辑  收藏  举报