zoukankan      html  css  js  c++  java
  • .Net下一个类型转换神器

    引言

    类型转换经常遇到,最常用的应该是string类型转换为其它基元类型,常见于http参数类型转换。Convert静态类的Convert.ChangeType()方法可以把实现IConvertible接口的类型转换为其它也实现这个接口的类型,也等同于里面的ToInt32()、ToDecimal()等方法的功能,但不支持转换到这类型的可空类型,因为Nullable<>类型并没有实现IConvertible这个接口;此外JavaScriptSerializer.ConvertToType()这个方法相对强大,支持转换到Nullable<>类型、字符串到枚举,也支持IDictionary<string,object>类型转换为对象,居于原因,是为了JavaScriptSerializer在反序列化JSON为动态类型后,能把这些类型进行转换为常见类型,但JavaScriptSerializer.ConvertToType方法不能扩展。

    神器来了

    名称:

    今天主角转换器叫Converter,这是花费我几天时间才琢磨出来。

    点再下载代码(也要点个赞哦)

    功能:

    今它支持基础类型、decimal、guid和枚举相互转换以及这些类型的可空类型和数组类型相互转换,支持字典和DynamicObject转换为对象以及字典和DynamicObject的数组转换为对象数组。

    扩展性:

    可以增加其它类型转换,或修改某种转换规则,也就是说转换器是支持功能部分重写和功能扩展的,功能移除也支持,比如想干掉DynamicObject类型的转换也可以。

    神器使用方法

    1、使用静态方法,Converter.Cast(object value,Type targetType)或其泛型方法 :Converter.Cast<int?[]>(new[] {null, "2" }

    2、实例化来调用,new Converter().Convert(new[] { null, "2" }, typeof(int?));

    Converter的扩展

    扩展是灵魂,如果没有扩展功能,这东东和JavaScriptSerializer.ConvertToType()有啥区别,Converter在扩展方面下了一翻功夫。转换器里包含多个转换单元,这些单元可以增加、删除、替换和调整顺序,自定义的单元,可以从已有单元派生,也可以完整实现,然后插入到转换器的转换单元管理器中,转换器就升级了。而转换单元本身也能调用转换器来转换,这就形成一种可扩展的递归。

    关系代码如下:

     /// <summary>
            /// 类型转换
            /// </summary>
            public Converter()
            {
                this.Items = new ContertItems()
                    .AddLast<NoConvert>()
                    .AddLast<NullConvert>()
                    .AddLast<PrimitiveContert>()
                    .AddLast<NullableConvert>()
                    .AddLast<DictionaryConvert>()
                    .AddLast<ArrayConvert>()
                    .AddLast<DynamicObjectConvert>();
            }
    转换器由转换单元组成
     /// <summary>
        /// 定义类型转换单元
        /// </summary>
        public interface IConvert
        {
            /// <summary>
            /// 将value转换为目标类型
            /// 并将转换所得的值放到result
            /// 如果不支持转换,则返回false
            /// </summary>
            /// <param name="converter">转换器实例</param>
            /// <param name="value">要转换的值</param>
            /// <param name="targetType">转换的目标类型</param>
            /// <param name="result">转换结果</param>
            /// <returns>如果不支持转换,则返回false</returns>
            bool Convert(Converter converter, object value, Type targetType, out object result);
        }
    转换单元可以调用转换器


    删除转换单元:

    var converter = new Converter();
    converter.Items.Remove<DynamicObjectConvert>();

    增加转换单元:

    编写一个单元,实现IConvert接口,比如 class JObjectConvert:IConvert{},然后就可以把这个转换单元添加到转换器实例中,

    var converter = new Converter();
    converter.Items.AddFrist<JObjectConvert>();

    转换单元重写:

    比如从string转换为decimal类时,要求5位小数,我们可以写一个单元,class MyPrimitiveContert:PrimitiveContert{},然后改写里面的Convert方法,再把MyPrimitiveContert替换转换器里的PrimitiveContert单元

    var converter = new Converter();
    converter.Items.Replace<PrimitiveContert, MyPrimitiveContert>();

    后记

    之所以写这个东西,是因为我的NetworkSocket组件里websocket部分很需要一个灵活的类型转换器,客户端发送{api:"login",parameters:[{account:"admin",password:"123456"}]}这样结构的Json给服务器,服务要解析这个json然后反射执行Login(Userinfo user)这个方法。

    这里parameters数组里内容的类型是根据api的名称而定的,而要解析到api的内容,又要知道整个json的结构,形成蛋和鸡。只有将json解析为动态类型,才能满足要求,而JavaScriptSerializer反序列化为动态类型时,实际上是json对象转换为字典,json数组转换为ArrayList,其它基本类型差不多一一对应,枚举类型看情况;如果使用Json.Net解析,json对象转换为JObject动态类型,json数组转换为JArray,一些基本类型一般会对应JValue。

    当把序列化和反序列化抽象或定义为接口的时候,JavaScriptSerializer.Converter方法就无力了,它没法转换JObject这些类型,也没有扩展的入口点,还有就是在.net core上也用不了。而Converter只需要添加一个单元转换单元,用来实现Json.Net里面几个动态类型转换为Api参数的类型,就可以达到完美的转换效果。

  • 相关阅读:
    高级开发必须理解的Java中SPI机制
    希尔排序--python
    SpringContextAware使用详解
    visio professional 2013 密钥
    二分查找--python
    [Oracle]单行字符函数
    [Oracle]sqlplus调整列宽
    [Oracle]MacOS sqlplus上下选择命令
    [Oracle]开启SCOTT账户
    [Oracle]Macos 安装Oracle Client 11g 11.2.0.4
  • 原文地址:https://www.cnblogs.com/kewei/p/5025722.html
Copyright © 2011-2022 走看看