引言

类型转换经常遇到,最常用的应该是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参数的类型,就可以达到完美的转换效果。

.Net下一个类型转换神器的更多相关文章

  1. Go将统治下一个10年?Go语言发展现状分析

    “本文是国内Go语言大中华区首席布道师——许式伟,在QCon2015上海站上的分享.他预测Go语言10年内一定会超过C和java,并且统治这一个10年. Go语言语法及标准库变化 Go从1.0版本到现 ...

  2. [LeetCode] Next Permutation 下一个排列

    Implement next permutation, which rearranges numbers into the lexicographically next greater permuta ...

  3. Javascript JQuery获取当前元素的兄弟元素/上一个/下一个元素(转)

    var chils= s.childNodes; //得到s的全部子节点 var par=s.parentNode; //得到s的父节点 var ns=s.nextSbiling; //获得s的下一个 ...

  4. SQL Server获取自增列下一个Id

    IDENT_CURRENT('TableName')为当前的最大标识值,IDENT_INCR('TableName')为设置的标识值增量, 两者相加即为下一个标识值 SELECT IDENT_CURR ...

  5. JMeter 将上一个请求的结果作为下一个请求的参数——使用正则提取器(转载)

    在接口测试和压力测试过程中,经常会将几个流程串联起来才能测试.如:我要进行获取用户信息接口测试,我就要先登录成功后,才能获取用户信息.所以,我就要首先要登录,获得我的登录凭证(tokenId或tick ...

  6. 获取当前时间UTC时间的下一个15分钟时间点

    ZonedDateTime zdt = ZonedDateTime.now(ZoneOffset.UTC); int now15Minute = zdt.getMinute() / P15MINUTE ...

  7. NASA的下一个十年(译)

    原文 MICHAEL ROSTON (New York Times) 从左起:木卫二:土卫六:经过火星的水手谷星的合成图:金星的拼接图 大多数人已经从人类第一次近距离看到冥王星的兴奋中冷静下来.下一个 ...

  8. SQL Server获取下一个编码字符串的实现方案分割和进位

        我在前一种解决方案SQL Server获取下一个编码字符实现和后一种解决方案SQL Server获取下一个编码字符实现继续重构与增强两篇博文中均提供了一种解决编码的方案,考虑良久对比以上两种方 ...

  9. SQL Server获取下一个编码字符实现继续重构与增强

        我在SQL Server获取下一个编码字符实现的博文中,虽然实现了这个问题,但是感觉维护起来比较麻烦,例如如果调整编码字符串的固定长度,就需要变更三个函数,这样的为何成本确实比较大.面向对象编 ...

随机推荐

  1. SQL动态列查询

    数据库中为了实现表格数据的自由设置,我们经常设计纵表,或者列定义的表(如下KeyValue),定义一个列超级多的表中每个字段的意义. 但是在设计时简单的东西却很容易被人们忘记,如下一个简单但是很松散的 ...

  2. SqlServer链接MySql操作步骤

    Sql Server版本 2008R2 1.从MySQL网站下载最新的MySQL ODBC驱动:http://www.mysql.com/downloads/connector/odbc/,我下载的版 ...

  3. Linux 性能优化工具包 sysstat 以及 sysstat 服务

    1. sysstat service 1.1. /etc/cron.d/sysstat Linux 下的性能分析,少不了要使用 sysstat 工具包中的命令.同时 sysstat 服务会自动每个10 ...

  4. CentOS系统在不重启的情况下为虚拟机添加新硬盘

    一.概述 用过虚拟机的都知道,如果在系统运行的时候去给虚拟机添加一块新设备,比如说硬盘,系统是读取不到这个新硬盘的,因为系统在启动的时候会去检测硬件设备.但是我们也可能会遇到这样的情况,比如正在运行比 ...

  5. Linux 下从头再走 GTK+-3.0 (二)

    仅仅创建一个空白窗口是不够的,下面我们为创建的窗口添加一个按钮. 以 Hello,World!为例. 首先创建一个源文件:example2.c 内容如下. #include <gtk/gtk.h ...

  6. 【转载】Android Metro风格的Launcher开发系列第二篇

    前言: 各位小伙伴们请原谅我隔了这么久才开始写这一系列的第二篇博客,没办法忙新产品发布,好了废话不说了,先回顾一下:在我的上一篇博客Android Metro风格的Launcher开发系列第一篇写了如 ...

  7. Java实现点击一个Jlabel增加一个Jlabel的小功能

    当界面生成以后,自己想做一个点击一个Jlabel增加一个Jlabel,即类似于QQ的添加好友以后可以及时的加进一个好友.自己做了好久,发现不能及时刷新.在网上查了一下,然后自己研究了一小会.发现需要v ...

  8. windows 7 语言切换 Vistalizator

    Try Vistalizator - this tool allows you to change display language in Windows editions other than (o ...

  9. codeforces118D. Caesar's Legions

    地址:http://www.codeforces.com/problemset/problem/118/D 题目: Gaius Julius Caesar, a famous general, lov ...

  10. form表单验证和事件、正则表达式

    1.表单验证<form></form> (1).非空验证(去空格) (2).对比验证(跟一个值对比) (3).范围验证(根据一个范围进行判断) (4).固定格式验证:电话号码, ...