前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >.NET Core 3.0 里新的JSON API

.NET Core 3.0 里新的JSON API

作者头像
solenovex
发布2019-10-25 15:56:18
2.1K0
发布2019-10-25 15:56:18
举报
文章被收录于专栏:草根专栏

为什么需要新的JSON API?

JSON.NET 大家都用过,老版本的ASP.NET Core也依赖于JSON.NET。

然而这个依赖就会引起一些版本问题:例如ASP.NET Core某个版本需要使用JSON.NET v10,而另一个库需要使用JSON.NET v11;或者JSON.NET 出现了一个新版本,而ASP.NET Core还不能支持这个版本,而您却想使用该版本。

System.Text.Json

随着NET Core 3.0的出现,出现了System.Text.Json命名空间和它下面一些用于处理JSON的类。

特点

这个内置JSON API具有与生俱来的高性能、地分配的特点:

JSON.NET 使用.NET 里面的字符串作为基本数据类型,其实也就是UTF16,而.NET Core中新的JSON API直接使用数据原始的UTF8格式。

新的JSON API基于Span<byte>这个数据类型来进行操作JSON数据,从而具有低分配的特点,这就可以极大的改善吞吐量和内存使用情况。

但是新的JSON API的特性还不那么丰富,有一些JSON.NET具有的特性都还不支持。

例子

随便找了一个JSON示例文件:

"post Title": "Programming" , 
"language": 
"author": 
"firstName": "Nick", 
"lastName": "Carter" 
"publishedAt 
"wordCount " : 
13435, 
"isoriginal"• 
true, 
"tags" • 
• "C#", "JSON API", ".NET Core" ]
"post Title": "Programming" , "language": "author": "firstName": "Nick", "lastName": "Carter" "publishedAt "wordCount " : 13435, "isoriginal"• true, "tags" • • "C#", "JSON API", ".NET Core" ]

针对这个文件,需要修改一下它的属性:

sample.json File P 「 operties 
Advanced 
Custom T00 一 Namespace 
Custom T00- 
Copy to Output 「 to 
Build Action 
〔 を
sample.json File P 「 operties Advanced Custom T00 一 Namespace Custom T00- Copy to Output 「 to Build Action 〔 を

Utf8JsonReader

先使用 Utf8JsonReader 来读取JSON文件。

Utf8JsonReader 并不会读取文件或者stream,它会读取Span数据类型。

直接上代码:

static void Elgin(stringC] 
gx•gs 
File . ReadAllBytes( path: "sample.json" ); 
var dataBytes 
dataBytes .AsSpan( ) ; 
var jsonSpan 
new Utf8JsonReader(jsonSpan); 
var jsonReader = 
while (jsonReader.Read()) 
Console . jsonReader)); 
I reference 
private static string GetTokenInfo(Utf8JsonReader jsonReader) 
jsonReader . TokenType switch 
JsonTokenType.StartObject 
"START OBJECT" , 
JsonTokenType 
. EndObject "END OBJECT" , 
JsonTokenType.StartArray 
"START ARRAY" , 
JsonTokenType. EndArray "End ARRAY" , 
JsonTokenType. PropertyName $ " PROPERTY: {jsonReader.GetString()}" , 
$"COMMENT: {jsonReader.GetString()}" , 
JsonTokenType 
. Comment 
JsonTokenType.String $"STRING: {jsonReader.GetString()}" , 
$"NUMBER: {jsonReader.GetInt32()}" , 
JsonTokenType.Number 
$"BOOL: {jsonReader.GetBoolean()}" , 
JsonTokenType 
. True 
$"BOOL: {jsonReader.GetBoolean()}" , 
JsonTokenType 
JsonTokenType.Null 
"NULL" , 
$ TOKEN: {jsonReader.TokenType}"
static void Elgin(stringC] gx•gs File . ReadAllBytes( path: "sample.json" ); var dataBytes dataBytes .AsSpan( ) ; var jsonSpan new Utf8JsonReader(jsonSpan); var jsonReader = while (jsonReader.Read()) Console . jsonReader)); I reference private static string GetTokenInfo(Utf8JsonReader jsonReader) jsonReader . TokenType switch JsonTokenType.StartObject "START OBJECT" , JsonTokenType . EndObject "END OBJECT" , JsonTokenType.StartArray "START ARRAY" , JsonTokenType. EndArray "End ARRAY" , JsonTokenType. PropertyName $ " PROPERTY: {jsonReader.GetString()}" , $"COMMENT: {jsonReader.GetString()}" , JsonTokenType . Comment JsonTokenType.String $"STRING: {jsonReader.GetString()}" , $"NUMBER: {jsonReader.GetInt32()}" , JsonTokenType.Number $"BOOL: {jsonReader.GetBoolean()}" , JsonTokenType . True $"BOOL: {jsonReader.GetBoolean()}" , JsonTokenType JsonTokenType.Null "NULL" , $ TOKEN: {jsonReader.TokenType}"

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 表达式。

运行程序

结果如下:

hema Selected> 
"post Title": "Programming" , 
"language": 
"author": 
"lastName": "Carter" 
"firstName": 
"publishedAt 
"Nick", 
"wordCount " : 
13435, 
"isoriginal"• 
true, 
"tags" • 
• "C#", "JSON 
API", ".NET Core" 
PROPERTY: postTit1e 
STRING: Programming 
PROPERTY: language 
STRING: 
PROPERTY: author 
START OBJECT 
PROPERTY: firstName 
STRING: Nick 
PROPERTY: lastName 
STRING: carter 
END OBJECT 
PROPERTY: publishedAt 
STRING: 
PROPERTY: wordcount 
NUMBER: 13435 
PROPERTY: isoriginal 
BOOL: True 
PROPERTY: tags 
START ARRAY 
STRING: 
STRING: JSON API 
STRING: .NET core 
End ARRAY 
END OBJECT
hema Selected> "post Title": "Programming" , "language": "author": "lastName": "Carter" "firstName": "publishedAt "Nick", "wordCount " : 13435, "isoriginal"• true, "tags" • • "C#", "JSON API", ".NET Core" PROPERTY: postTit1e STRING: Programming PROPERTY: language STRING: PROPERTY: author START OBJECT PROPERTY: firstName STRING: Nick PROPERTY: lastName STRING: carter END OBJECT PROPERTY: publishedAt STRING: PROPERTY: wordcount NUMBER: 13435 PROPERTY: isoriginal BOOL: True PROPERTY: tags START ARRAY STRING: STRING: JSON API STRING: .NET core End ARRAY END OBJECT

可以看到sample.json文件里面的每个Token都被正确的显示了。

JsonDocument类

JsonDocument是基于Utf8JsonReader 构建的。JsonDocument 可分析 JSON 数据并生成只读文档对象模型 (DOM),可对模型进行查询,以支持随机访问和枚举。使用 JsonDocument 分析常规 JSON 有效负载并访问其所有成员比使用 Json.NET 快 2-3 倍,且为合理大小(即 < 1 MB)的数据所分配的量非常少。

JsonDocument可以处理Span,也可以处理Stream。

例子:

using 
using 
var 
var 
File .0penRead( path: 
"sample.json"); 
stream = 
JsonDocument . Parse( stream) ; 
doc
using using var var File .0penRead( path: "sample.json"); stream = JsonDocument . Parse( stream) ; doc

这里我通过File.OpenRead把json文件转化为stream。然后使用JsonDocument.Parse方法把stream解析成JSON文档对象模型。

注意,这里我使用了C# 8的using var语法,这个以后再说。

下面我们开始从这个JSON文档对象模型的根节点开始遍历,也就是RootElement:

root 
var 
doc . Root Element;
root var doc . Root Element;

然后通过root这个JsonElement类型的对象的GetProperty方法来获得相应的属性,而且这个方法可以连串使用:

firstName 
var 
root .GetProperty( "author") 
. GetProperty( "firstName" ) 
. GetString( ) ;
firstName var root .GetProperty( "author") . GetProperty( "firstName" ) . GetString( ) ;

最后一行使用GetString方法来获得该属性的字符串值。

然后我们可以写一个递归调用的方法来遍历整个模型的每个属性:

EnumerateElement(JsonElement root) 
private static void 
foreach (var prop in root. EnumerateObject()) 
if (prop.Value.ValueKind = 
JsonValueKind .0bject) 
Console . WriteLine($" {prop . Name}" ) ; 
Console.WriteLine(" 
- Object Start - 
EnumerateElement(prop . Value); 
Console.WriteLine(" 
Object End 
else 
{prop. Value.GetRawText()}");
EnumerateElement(JsonElement root) private static void foreach (var prop in root. EnumerateObject()) if (prop.Value.ValueKind = JsonValueKind .0bject) Console . WriteLine($" {prop . Name}" ) ; Console.WriteLine(" - Object Start - EnumerateElement(prop . Value); Console.WriteLine(" Object End else {prop. Value.GetRawText()}");

这个方法接受JsonElement类型的对象,然后对该元素的属性进行循环。

如果当前属性是另一个对象,那么就继续递归调用这个方法;

否则就输出原始的文本。

最后调用该方法:

File. OpenRead( 
"sample.json"); 
using var stream 
path: 
JsonDocument . Parse( stream) ; 
using var doc - 
doc . Root Element; 
var root 
var firstName 
root .GetProperty( "author") 
. GetProperty( "firstName" ) 
. GetString( ) 
Console.WriteLine($"First Name is {firstName}"); 
EnumerateElement(root);
File. OpenRead( "sample.json"); using var stream path: JsonDocument . Parse( stream) ; using var doc - doc . Root Element; var root var firstName root .GetProperty( "author") . GetProperty( "firstName" ) . GetString( ) Console.WriteLine($"First Name is {firstName}"); EnumerateElement(root);

输出结果为:

First Name 
post Title: 
language: 
author 
is Nick 
" Programmi ng" 
Object Start 
irstName: "Nick" 
lastName: "Carter" 
Object End 
publishedAt : 
22T2ø: 21 . eeøz" 
rdcount: 13435 
isoriginal: true 
ags: [ "C#", " 
JSON API 
% ".NET core" :
First Name post Title: language: author is Nick " Programmi ng" Object Start irstName: "Nick" lastName: "Carter" Object End publishedAt : 22T2ø: 21 . eeøz" rdcount: 13435 isoriginal: true ags: [ "C#", " JSON API % ".NET core" :

与json文件的内容匹配。

Utf8JsonWriter类

下面研究一下如何写入json文件。这里需要使用Utf8JsonWriter类。

直接看代码:

这个类需要传递的参数类型是Stream或者Buffer,也就是向Stream或Buffer里面写入数据。

那么就提供一个buffer:

var buffer = 
new 
= new Utf8JsonWriter(buffer); 
using var 
'son
var buffer = new = new Utf8JsonWriter(buffer); using var 'son

下面单独写一个方法,来生成json数据:

PopulateJson(Utf8JsonWriter json) 
private static void 
Json. 
write 
W rite 
W r ite Base64St ringVaIue 
W r ite Boolean 
W r ite BooleanVaIue 
W r iteCo m mentVaIue 
Write EndArray 
W r ite EndObject 
Wr iteNuII 
W r ite NullVaIue 
Write Number 
W rite Nu mbe rVaIue 
void 
void 
void 
void 
void 
void 
void 
void 
void 
void 
void 
(JsonEncodedText propertyName, ReadOnIyS 
Writes the pre-encoded property name and 
a name/value pair of a JSON object. 
utf8PropertyName, Re; 
propertyName, ReadO 
(string propertyName,
PopulateJson(Utf8JsonWriter json) private static void Json. write W rite W r ite Base64St ringVaIue W r ite Boolean W r ite BooleanVaIue W r iteCo m mentVaIue Write EndArray W r ite EndObject Wr iteNuII W r ite NullVaIue Write Number W rite Nu mbe rVaIue void void void void void void void void void void void (JsonEncodedText propertyName, ReadOnIyS Writes the pre-encoded property name and a name/value pair of a JSON object. utf8PropertyName, Re; propertyName, ReadO (string propertyName,

参数类型是Utf8JsonWriter。通过智能提示可以看到它提供了很多用于写入不同类型数据的方法。

写JSON对象

现在我想写一个json对象,那么就从WriteStartObject()开始,然后以WriteEndObject()结束:

json . WriteStartObject( ); 
json.WriteEndObject();
json . WriteStartObject( ); json.WriteEndObject();

这样的话,实际上我已经拥有了一个合法的json文档。

写属性和值

可以分开写属性和值:

json . WritePropertyName( "title"); 
json . WriteStringValue( " Programming" ) ;
json . WritePropertyName( "title"); json . WriteStringValue( " Programming" ) ;

也可以同时把属性和值写出来:

json .WriteString( propertyName: "language" , 
"C#"); 
value:
json .WriteString( propertyName: "language" , "C#"); value:

显示JSON数据

我先写这些内容,然后在Main方法里面调用一下:

grgs) 
static void Mgin(stringC] 
new ArrayBufferWriter<byte>(); 
var buffer - 
new Utf8JsonWriter(buffer); 
using var Ison = 
PopulateJson( j son) ; 
I reference 
private static void PopulateJson(Utf8JsonWriter json) 
Json. 
Json. 
Json. 
Json. 
Json. 
Json. 
Json. 
WriteStartObject( ) ; 
WritePropertyName( "title"); 
WriteStringValue( " Programming" ) ; 
WriteString( propertyName: "language" , 
WriteString( propertyName: "firstName" , 
WriteString( propertyName: "lastName" , 
WriteEndObject( ) ; 
"C#"); 
value: 
"Nick"); 
value: 
"Carter"); 
value:
grgs) static void Mgin(stringC] new ArrayBufferWriter<byte>(); var buffer - new Utf8JsonWriter(buffer); using var Ison = PopulateJson( j son) ; I reference private static void PopulateJson(Utf8JsonWriter json) Json. Json. Json. Json. Json. Json. Json. WriteStartObject( ) ; WritePropertyName( "title"); WriteStringValue( " Programming" ) ; WriteString( propertyName: "language" , WriteString( propertyName: "firstName" , WriteString( propertyName: "lastName" , WriteEndObject( ) ; "C#"); value: "Nick"); value: "Carter"); value:

首先需要告诉writer把它的内容flush给buffer,使用这个buffer我们可以获得writer的输出,这样的话就会得到一个byte数组,然后把这个byte数组转化为字符串,这样就可以在控制台显示它了:

new ArrayBufferWriter<byte>(); 
var buffer = 
new Utf8JsonWriter(buffer); 
using var Ison = 
PopulateJson( json); 
json. Flush(); 
var output = buffer.WrittenSpan.ToArray(); 
Encoding. UTF8. GetString(output); 
var outJson - 
Console. WriteLine(outJson);
new ArrayBufferWriter<byte>(); var buffer = new Utf8JsonWriter(buffer); using var Ison = PopulateJson( json); json. Flush(); var output = buffer.WrittenSpan.ToArray(); Encoding. UTF8. GetString(output); var outJson - Console. WriteLine(outJson);

运行一下看看效果:

{"title" : "Programming" , "language" : "C#" , "firstName" : "Nick" , "lastName" : "Carter"}
{"title" : "Programming" , "language" : "C#" , "firstName" : "Nick" , "lastName" : "Carter"}

没啥太大的问题,就是格式不好看。

对输出进行格式化

.NET Core提供了一个JsonWriterOptions类,它可以对Writer进行一些设置。

var options = 
Indented = 
new JsonWriterOptions 
true 
new Utf8JsonWriter(buffer, 
using var Ison = 
options);
var options = Indented = new JsonWriterOptions true new Utf8JsonWriter(buffer, using var Ison = options);

这里对输出进行了缩进,最后把这个options传递给Utf8JsonWriter的构造函数即可。

再次运行:

"title": "Programming", 
"language" • " 
"firstName": "Nick", 
"lastName" : "Carter"
"title": "Programming", "language" • " "firstName": "Nick", "lastName" : "Carter"

现在好看多了。

JsonSerializer

前面几节的内容可能稍微有点底层,我们大部分时候可能只需要对C#的类进行串行化或者将JSON数据反串行化成C#类,在.NET Core 3.0里面,我们可以使用JsonSerializer这个类来做这些事情。

例子:

还是使用之前用到的json数据:

"post Title": "Programming" 
"language": 
"author": 
"firstName": "Nick", 
"lastName": "Carter" 
"publishedAt 
"wordCount " : 
13435, 
"isoriginal"• 
true, 
"tags" • 
"JSON API", ".NET Core"
"post Title": "Programming" "language": "author": "firstName": "Nick", "lastName": "Carter" "publishedAt "wordCount " : 13435, "isoriginal"• true, "tags" • "JSON API", ".NET Core"

然后我们需要建建立两个类,对应这个文件:

public class 
BlogPost 
O references 
public 
O references 
public 
O references 
public 
O references 
public 
O references 
public 
O references 
public 
public 
I reference 
{ get; set; } 
string 
Post Title 
{ get; set; } 
string 
Language 
{ get; set; } 
Author Author 
{ get; set; 
DateTime PublishedAt 
{ get; set; } 
int 
WordCount 
{ get; set; } 
bool 
Isoriginal 
:iag$ 
{ get; set; } 
stringC] 
public class Author 
O references 
{ get; set; } 
public string 
FirstName 
O references 
{ get; set; } 
public string 
LastName
public class BlogPost O references public O references public O references public O references public O references public O references public public I reference { get; set; } string Post Title { get; set; } string Language { get; set; } Author Author { get; set; DateTime PublishedAt { get; set; } int WordCount { get; set; } bool Isoriginal :iag$ { get; set; } stringC] public class Author O references { get; set; } public string FirstName O references { get; set; } public string LastName

反串行化

可以使用JsonSerializer类的Deserialize()方法对json数据反串行化。这个方法支持三种类型的输入参数,分别是:

  • JSON数据的字符串
  • Utf8JsonReader
  • ReadOnlySpan<byte>,它里面包含JSON数据

为了简单一点,我直接把json文件读取成字符串,然后传给Deserialize方法:

var text = 
var post = 
File. ReadAllText( path: 
"sample. json"); 
JsonSerializer . text); 
Console . WriteLine(post. PostTitle); 
Console . WriteLine($" {post .Author . FirstName} 
{post .Author . LastName}"); 
Console . WriteLine( post. PublishedAt);
var text = var post = File. ReadAllText( path: "sample. json"); JsonSerializer . text); Console . WriteLine(post. PostTitle); Console . WriteLine($" {post .Author . FirstName} {post .Author . LastName}"); Console . WriteLine( post. PublishedAt);

然后我试图打印出反串行化之后的一些属性数据。但是这不会成功。因为JSON文件里面数据的大小写命名规范使用的是camel casing(简单理解为首字母是小写的),而默认情况下Deserializer会寻找Pascal casing这种规范(简单理解为每个单词的首字母都是大写的)的属性名。

格式化

为解决这个问题,就需要使用JsonSerializerOptions类:

var 
var 
options = 
new JsonSeria1izerOptions 
PropertyNamingPolicy = 
JsonNamingPolicy . CamelCase 
post = 
JsonSerializer.Deserialize<BlogPost>(text, options);
var var options = new JsonSeria1izerOptions PropertyNamingPolicy = JsonNamingPolicy . CamelCase post = JsonSerializer.Deserialize<BlogPost>(text, options);

建立该类的一个实例,设置PropertyNamingPolicy为CamelCase,然后把这个实例传递给Deserialize方法的第二个参数。

运行看结果:

Programmi ng 
Nick - Carter 
18/22/2819 PM
Programmi ng Nick - Carter 18/22/2819 PM

这次就没有问题了。

串行化

JsonSerializer也支持串行化,也就是把C#数据转化为JSON数据:

这里使用了相同的options。

运行结果:

如果想让输出结果更好看一些,可以在JsonSerializerOptions里面进行相应的设置:

这次输出结果为:

总结

总结一下.NET Core 3.0新的JSON API:

  • Utf8JsonReader - 读操作,快速,低级
  • Utf8JsonWriter - 写操作,快速,低级
  • JsonDocument - 基于DOM,快速
  • JsonSeriliazer - 串行化/反串行化,快速

另外 JSON.NET 仍然被支持。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2019-10-25 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 为什么需要新的JSON API?
  • System.Text.Json
    • 特点
      • 例子
      • Utf8JsonReader
      • JsonDocument类
      • Utf8JsonWriter类
        • 写JSON对象
          • 写属性和值
            • 显示JSON数据
              • 对输出进行格式化
              • JsonSerializer
                • 反串行化
                  • 格式化
                    • 串行化
                    • 总结
                    领券
                    问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档