zoukankan      html  css  js  c++  java
  • [置顶] .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,欢迎交流。

  • 相关阅读:
    UML基础—结构和组成
    Hadoop完全分布式
    Hadoop完全分布式
    查看端口是否启用
    查看端口是否启用
    hadoop namenode启动失败
    hadoop namenode启动失败
    Hadoop2.7.5伪分布式安装
    Hadoop2.7.5伪分布式安装
    给用户添加sudo权限
  • 原文地址:https://www.cnblogs.com/pangblog/p/3278289.html
Copyright © 2011-2022 走看看