JSON.NET 你们都用过,老版本的ASP.NET Core也依赖于JSON.NET。 json
然而这个依赖就会引发一些版本问题:例如ASP.NET Core某个版本须要使用JSON.NET v10,而另外一个库须要使用JSON.NET v11;或者JSON.NET 出现了一个新版本,而ASP.NET Core还不能支持这个版本,而您却想使用该版本。 数组
随着NET Core 3.0的出现,出现了System.Text.Json命名空间和它下面一些用于处理JSON的类。 函数
这个内置JSON API具备与生俱来的高性能、低分配的特色: post
JSON.NET 使用.NET 里面的字符串做为基本数据类型,其实也就是UTF16,而.NET Core中新的JSON API直接使用数据原始的UTF8格式。 性能
新的JSON API基于Span<byte>这个数据类型来进行操做JSON数据,从而具备低分配的特色,这就能够极大的改善吞吐量和内存使用状况。 ui
可是新的JSON API的特性还不那么丰富,有一些JSON.NET具备的特性都还不支持。 spa
随便找了一个JSON示例文件: code
针对这个文件,须要修改一下它的属性: orm
先使用 Utf8JsonReader 来读取JSON文件。 xml
Utf8JsonReader 并不会读取文件或者stream,它会读取Span数据类型。
直接上代码:
Main方法里面,咱们使用File.ReadAllBytes从sample.json文件读取数格式为byte[],而后经过AsSpan这个扩展方法将其转化为Span<byte>数据类型,而后把它传递到 Utf8JsonReader 的构造函数来建立一个JSON的reader。
接下来使用while循环对JSON数据的每一个Token进行读取,每次执行Read()方法时,reader就会移动到JSON数据里面的下一个Token那里。
Token分红几种类型,GetTokenInfo方法就是判断一下Token的类型,并返回一些描述性信息,这里面应该是包含了全部的类型。这里面使用到了C# 8 的 switch 表达式。
运行程序
结果以下:
能够看到sample.json文件里面的每一个Token都被正确的显示了。
JsonDocument是基于Utf8JsonReader 构建的。JsonDocument 可分析 JSON 数据并生成只读文档对象模型 (DOM),可对模型进行查询,以支持随机访问和枚举。使用 JsonDocument 分析常规 JSON 有效负载并访问其全部成员比使用 Json.NET 快 2-3 倍,且为合理大小(即 < 1 MB)的数据所分配的量很是少。
JsonDocument能够处理Span,也能够处理Stream。
例子:
这里我经过File.OpenRead把json文件转化为stream。而后使用JsonDocument.Parse方法把stream解析成JSON文档对象模型。
注意,这里我使用了C# 8的using var语法,这个之后再说。
下面咱们开始从这个JSON文档对象模型的根节点开始遍历,也就是RootElement:
而后经过root这个JsonElement类型的对象的GetProperty方法来得到相应的属性,并且这个方法能够连串使用:
最后一行使用GetString方法来得到该属性的字符串值。
而后咱们能够写一个递归调用的方法来遍历整个模型的每一个属性:
这个方法接受JsonElement类型的对象,而后对该元素的属性进行循环。
若是当前属性是另外一个对象,那么就继续递归调用这个方法;
不然就输出原始的文本。
最后调用该方法:
输出结果为:
与json文件的内容匹配。
下面研究一下如何写入json文件。这里须要使用Utf8JsonWriter类。
直接看代码:
这个类须要传递的参数类型是Stream或者Buffer,也就是向Stream或Buffer里面写入数据。
那么就提供一个buffer:
下面单独写一个方法,来生成json数据:
参数类型是Utf8JsonWriter。经过智能提示能够看到它提供了不少用于写入不一样类型数据的方法。
如今我想写一个json对象,那么就从WriteStartObject()开始,而后以WriteEndObject()结束:
这样的话,实际上我已经拥有了一个合法的json文档。
能够分开写属性和值:
也能够同时把属性和值写出来:
我先写这些内容,而后在Main方法里面调用一下:
首先须要告诉writer把它的内容flush给buffer,使用这个buffer咱们能够得到writer的输出,这样的话就会获得一个byte数组,而后把这个byte数组转化为字符串,这样就能够在控制台显示它了:
运行一下看看效果:
没啥太大的问题,就是格式很差看。
.NET Core提供了一个JsonWriterOptions类,它能够对Writer进行一些设置。
这里对输出进行了缩进,最后把这个options传递给Utf8JsonWriter的构造函数便可。
再次运行:
如今好看多了。
前面几节的内容可能稍微有点底层,咱们大部分时候可能只须要对C#的类进行串行化或者将JSON数据反串行化成C#类,在.NET Core 3.0里面,咱们可使用JsonSerializer这个类来作这些事情。
例子:
仍是使用以前用到的json数据:
而后咱们须要建创建两个类,对应这个文件:
可使用JsonSerializer类的Deserialize()方法对json数据反串行化。这个方法支持三种类型的输入参数,分别是:
JSON数据的字符串
Utf8JsonReader
ReadOnlySpan<byte>,它里面包含JSON数据
为了简单一点,我直接把json文件读取成字符串,而后传给Deserialize方法:
而后我试图打印出反串行化以后的一些属性数据。可是这不会成功。由于JSON文件里面数据的大小写命名规范使用的是camel casing(简单理解为首字母是小写的),而默认状况下Deserializer会寻找Pascal casing这种规范(简单理解为每一个单词的首字母都是大写的)的属性名。
为解决这个问题,就须要使用JsonSerializerOptions类:
创建该类的一个实例,设置PropertyNamingPolicy为CamelCase,而后把这个实例传递给Deserialize方法的第二个参数。
运行看结果:
此次就没有问题了。
JsonSerializer也支持串行化,也就是把C#数据转化为JSON数据:
这里使用了相同的options。
运行结果:
若是想让输出结果更好看一些,能够在JsonSerializerOptions里面进行相应的设置:
此次输出结果为:
总结一下.NET Core 3.0新的JSON API:
Utf8JsonReader - 读操做,快速,低级
Utf8JsonWriter - 写操做,快速,低级
JsonDocument - 基于DOM,快速
JsonSeriliazer - 串行化/反串行化,快速
另外 JSON.NET 仍然被支持。