去年,我在一篇文章用原始方法解析复杂字符串,json一定要用JsonMapper么?中介绍了简单的JSON解析的问题,那种方法在当时的环境是非常方便的,因为不需要生成实体类,结构很容易解析。但随着业务的变化,也会碰到超级变态的JSON,如果还按照以前的思路,会把人搞抽风掉,一旦结构变化,又要重来。所以今天给大家介绍一个简单的方法,轻轻松松搞定超级变态的JSON,虽然需要生成实体类。它就是开源的:JSON C# Class Generator组件。
.NET开源目录:【目录】本博客其他.NET开源项目文章目录
本文原文地址:http://www.cnblogs.com/asxinyu/p/dotnet_Opensource_project_json_generator.html
1.复杂的JSON啥样子?
看看下面这个图,文本文件都是12K,嵌套多层,说实话,我是没耐心看下去的。所以找了个工具结构化了一下。看图对比:
2.使用JSON C# Class Generator介绍和使用
JSON C# Class Generator是一个从JSON文本中生成C#内的应用程序。项目地址:https://jsonclassgenerator.codeplex.com/ ,目前支持数组,对象,整形,单精度,布尔,字符串和空值符类型。
使用过程很简单,如下图官网的截图所示:输入命名空间,主类名,生成的cs文件目录,以及类型即可。同时右边还有可见性,和一些简单的配置,简单易懂。
下面看看我们演示的这个JSON的使用,是在网络找到的一段演示JSON代码,我自己用到的实际数据有些比这还复杂。首先我们把JSON复制到JSON C# Class Generator中,设置相关参数,如下图所示,生成文件后,复制到项目中,下面我们可以写测试代码,看看起解析的格式:
说明一下,我们JSON反序列需要用到Newtonsoft组件,这个相信大部分都肯定比我熟悉。将生成的文件复制到项目中,并添加的引用,如下代码就反序列化OK了。是不是So Easy?
1
2
3
4
|
//获取jsonzifc
var
json = File.ReadAllText(
"复杂JSON例子.txt"
);
//使用Newtonsoft反序列化
var
model = JsonConvert.DeserializeObject<TestJson>(json);
|
其实这个套路大部分人也都懂,但我知道的也有一些在线解析工具,但貌似遇到特别复杂的,不好整,这个工具是我见过的里面比较简单易用。看看解析的数据,结构化后,非常清晰,自己想要那个数据,自己去属性依次获取就好了。
3.就这么完了?No,没那么简单
工具能解决的问题一般不是全部,能解决8成就很不错了,虽然这个工具刚开始就解决了很多问题,但随着JSON的复杂化,也碰到了一些坑,看看如何填坑。现在已经是填坑专业户了。
3.1 为什么要使用嵌套类(nested classes)
为什么在对于复杂的类型我选择使用嵌套类,原因很简单,结构更清晰,否则生成的cs文件里面一堆类,都搞不清楚那个是最外面的,对这种需求,适当的代码冗余就无所谓了,我需要的是快速的拿到自己想要的数据。何况是自动生成的。当然对于简单的类型,不使用嵌套类也是可以的,看个人习惯吧。
3.2 解析会碰到类型名称为数字的字段
虽然比较奇葩,但的确是遇到了,获取的JSON字符串里面的字段名称为数字,怎么破?鬼知道人家这么拼出来的,反正是为了采集,还得有解药才行。既然原始的是数字,那就把原始的给改了,我们把属性是数字的地方,都给加一个默认值,然后生成实体类,不就可以了,不过想在这么一堆乱七八糟的东西里面把所有的数字属性加个默认值,也不容易啊,还好有万能的正则表达式。
我们写一个简单的正则表达式匹配方法,把数字属性统一替换为前面加个默认的字母A,代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
|
public
static
string
GetNewJson(String json)
{
Regex reg =
new
Regex(
"\"([^\"]d*)\":"
);
//注意里面的引号 要用双引号表示,而不是用反斜杠
MatchEvaluator matchEval =
new
MatchEvaluator(ReplaceStar);
return
reg.Replace(json, matchEval);
}
static
string
ReplaceStar(Match match)
{
string
str = match.Value.Replace(
"\""
,
""
).Replace(
":"
,
""
);
return
"\"A"
+ str +
"\":"
;
}
|
3.3 相同结构的,但解析为不同的类名
很多时候某个属性下面有多个相同结构的对象,但是会根据名称解析为不同的类,结构基本一直,在获取对象数据的时候,又不是数组对象,循环还不好搞,那怎么破?
手动建一个中间类,结构和他们一样,使用TinyMaper,这里有文章介绍,将这些类型默认都映射到这个中间类中,然后其他各个相同结构对象的解析只需要写一份代码就OK了。看看下面的方法:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
public
class
LineRato
{
public
string
LgMailNo;
public
string
CreateTime;
public
string
EventTime;
public
string
EventAddress;
public
string
NodeType;
public
string
Timeout;
/// <summary>将其他结构相同的类转换为中间类</summary>
public
static
LineRato GetLine<T>(T model)
{
TinyMapper.Bind<T, LineRato>();
return
TinyMapper.Map<LineRato>(model);
}
}
|
3.4 空值导致实体类字段缺失
某些时候我们在第一次获取JSON值的时候,由于某些值是空的,导致实际的JSON值里面没包括该字段,所以在生成的时候实体类里面肯定没有这个字段,而实际多次后发现某些情况下该值又不为空,会导致解析失败,这个时候怎么破?
没办法破了,为了省事,看看是那里缺少字段,手动加上吧,就一行代码的事情。
总之,在解决了80%问题后,剩下的方法比问题多多了,大家各显神通吧。这里只是吧自己的解决过程写下来,最快的解决问题,OK!其他费脑细胞的事情还是给有精力的人!
4.资源
本文代码:JsonClassTest.rar
项目地址:https://jsonclassgenerator.codeplex.com/