今天在盒子闲逛,无意中看到有人说XE7自带的Json对象序列化很慢,帖子在这里:http://bbs.2ccc.com/topic.asp?topicid=464378;经过测试的确如此。
     但是 D10.2后,自带的 Json 做了优化,性能大大的提高了100多倍。

和其他json库对比了序列化和反序列化性能,JsonDataObjects 性能最好,但是只支持简单的对象,不支持结构体,QJson 则不支持动态数组,不支持 Attributes (RTTI),比如需要排除某些字段,System.JSON 和 XSuperObject 可支持。总体来说,个人比较喜欢10.2新增的TJsonSerializer,使用方便,无需第三方库。

全部代码如下:

unit Unit1;
{$DEFINE XSuperObject}
{$DEFINE QJson }
{$DEFINE JsonDataObjects} interface uses
Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics, Vcl.Controls,
Vcl.Forms, Vcl.Dialogs, Vcl.StdCtrls, Vcl.Samples.Spin
{$IF CompilerVersion>31.0}, System.JSON.Types, System.JSON.Serializers{$ENDIF}; type {$IF CompilerVersion>31.0}
[JsonSerialize(TJsonMemberSerialization.&Public)]
{$ENDIF} TObj1 = class
private
F_i: Integer;
f_d: TDateTime;
f_s: string;
// f_a: TArray<string>;
public
constructor Create;
published
property field_s: string read f_s write f_s;
property field_i: Integer read F_i write F_i;
property field_d: TDateTime read f_d write f_d;
// property field_a: TArray<string> read f_a write f_a;
end; TForm1 = class(TForm)
btnObjectToJsonString: TButton;
btnJsonSerializer: TButton;
mmoLog: TMemo;
seTestNumber: TSpinEdit;
lbl1: TLabel;
btnXSuperObject: TButton;
btnQJson: TButton;
btnJsonDataObjects: TButton;
btnParseFile: TButton;
dlgOpen1: TOpenDialog;
procedure btnObjectToJsonStringClick(Sender: TObject);
procedure btnJsonSerializerClick(Sender: TObject);
procedure FormCreate(Sender: TObject);
procedure btnXSuperObjectClick(Sender: TObject);
procedure btnQJsonClick(Sender: TObject);
procedure btnJsonDataObjectsClick(Sender: TObject);
procedure btnParseFileClick(Sender: TObject);
private
procedure Log(const S: string);
public
{ Public declarations }
end; var
Form1: TForm1;
TestNumber: Integer; implementation uses
System.Diagnostics,
System.IOUtils,
REST.JSON,
System.JSON,
{$IFDEF JsonDataObjects}
JsonDataObjects,
{$ENDIF}
{$IFDEF XSuperObject}
XSuperObject,
{$ENDIF}
{$IFDEF QJson}
qjson,
{$ENDIF}
rtcMW.System.JSONHelper; {$R *.dfm} function GetFileSize(const AFileName: string): Int64;
var
AttributeData: TWin32FileAttributeData;
begin
if not GetFileAttributesEx(PChar(AFileName), GetFileExInfoStandard, @AttributeData) then
RaiseLastOSError;
Int64Rec(Result).Lo := AttributeData.nFileSizeLow;
Int64Rec(Result).Hi := AttributeData.nFileSizeHigh;
end; procedure TForm1.btnObjectToJsonStringClick(Sender: TObject);
var
Foo: TObj1;
I: Integer;
sw: TStopwatch;
JSON: string;
begin
TestNumber := seTestNumber.Value; sw := TStopwatch.StartNew;
for I := to TestNumber - do
begin
Foo := TObj1.Create;
try
Foo.field_s := 'Hello World';
Foo.field_i := ;
Foo.field_d := Now;
JSON := REST.JSON.TJson.ObjectToJsonString(Foo);
finally
Foo.Free;
end;
end;
Log('TJson.ObjectToJsonString:' + sw.ElapsedMilliseconds.ToString + ' ms ' + JSON); sw := TStopwatch.StartNew;
for I := to TestNumber - do
begin
Foo := REST.JSON.TJson.JsonToObject<TObj1>(JSON);
try
finally
Foo.Free;
end;
end;
Log('TJson.ObjectToJsonString:' + sw.ElapsedMilliseconds.ToString + ' ms');
Log('=======================');
end; procedure TForm1.btnXSuperObjectClick(Sender: TObject);
{$IFDEF XSuperObject}
var
Foo: TObj1;
I: Integer;
sw: TStopwatch;
AJson: string;
{$ENDIF}
begin
{$IFDEF XSuperObject}
TestNumber := seTestNumber.Value;
sw := TStopwatch.StartNew; for I := to TestNumber - do
begin
Foo := TObj1.Create;
try
Foo.field_s := 'Hello World';
Foo.field_i := ;
Foo.field_d := Now;
AJson := XSuperObject.TJson.SuperObject<TObj1>(Foo).AsJSON();
finally
Foo.Free;
end; end;
Log('XSuperObject.TJson.SuperObject:' + sw.ElapsedMilliseconds.ToString + ' ms ' + AJson); for I := to TestNumber - do
begin
Foo := XSuperObject.TJson.Parse<TObj1>(AJson);
try
finally
Foo.Free;
end; end;
Log('XSuperObject.TJson.Parse:' + sw.ElapsedMilliseconds.ToString + ' ms');
Log('=======================');
{$ENDIF} end; procedure TForm1.btnJsonDataObjectsClick(Sender: TObject);
{$IFDEF JsonDataObjects}
var
Foo: TObj1;
I: Integer;
sw: TStopwatch;
AJson: string;
JsonObj: JsonDataObjects.TJsonObject;
{$ENDIF}
begin
{$IFDEF JsonDataObjects}
TestNumber := seTestNumber.Value;
sw := TStopwatch.StartNew; JsonObj := JsonDataObjects.TJsonObject.Create;
for I := to TestNumber - do
begin
Foo := TObj1.Create;
try
Foo.field_s := 'Hello World';
Foo.field_i := ;
Foo.field_d := Now;
JsonObj.FromSimpleObject(Foo);
AJson := JsonObj.ToJSON();
finally
Foo.Free;
end; end;
Log('JsonDataObjects.FromSimpleObject:' + sw.ElapsedMilliseconds.ToString + ' ms ' + AJson);
JsonObj.Free; sw := TStopwatch.StartNew;
for I := to TestNumber - do
begin
JsonObj := JsonDataObjects.TJsonObject.Parse(AJson) as JsonDataObjects.TJsonObject;
Foo := TObj1.Create;
try
JsonObj.ToSimpleObject(Foo);
finally
Foo.Free;
JsonObj.Free;
end;
end;
Log('JsonDataObjects.ToSimpleObject:' + sw.ElapsedMilliseconds.ToString + ' ms');
Log('=======================');
{$ENDIF}
end; procedure TForm1.btnParseFileClick(Sender: TObject); var
JsonObj: System.JSON.TJsonObject;
JsonObj1: JsonDataObjects.TJsonObject;
{$IFDEF QJson}
JsonObj2: TQJson;
{$ENDIF}
sw: TStopwatch;
FileName: string;
begin
if not dlgOpen1.Execute() then
Exit;
FileName := dlgOpen1.FileName;
Log(Format('%s,FileSize:%d Byte', [FileName, GetFileSize(FileName)]));
Log('======================='); sw := TStopwatch.StartNew;
JsonObj := System.JSON.TJsonObject.LoadFromFile(FileName).AsJsonObject;
try
if JsonObj = nil then
raise Exception.CreateFmt('%s 不是有效的 JSON 文件', [FileName]);
// Log(JsonObj.GetValue<string>('meta[0].field_name'));
Log('System.Json.TJsonObject.LoadFromFile:' + sw.ElapsedMilliseconds.ToString + ' ms');
finally
JsonObj.Free;
end;
{$IFDEF JsonDataObjects}
sw := TStopwatch.StartNew;
JsonObj1 := JsonDataObjects.TJsonObject.Create;
try
JsonObj1.LoadFromFile(FileName);
if JsonObj1 <> nil then
begin
Log('JsonDataObjects.TJsonObject.LoadFromFile:' + sw.ElapsedMilliseconds.ToString + ' ms');
end;
finally
JsonObj1.Free;
end;
{$ENDIF}
{$IFDEF QJson}
sw := TStopwatch.StartNew;
JsonObj2 := TQJson.Create;
try
JsonObj2.LoadFromFile(FileName);
if JsonObj2 <> nil then
begin
// Log(Json.ItemByPath('meta[0].field_name').AsString);
Log('TQJson.LoadFromFile:' + sw.ElapsedMilliseconds.ToString + ' ms');
end;
finally
JsonObj2.Free;
end;
{$ENDIF} end; procedure TForm1.btnJsonSerializerClick(Sender: TObject);
{$IF CompilerVersion>31.0}
var
Foo: TObj1;
I: Integer;
sw: TStopwatch;
AJson: string;
Serializer: TJsonSerializer;
{$ENDIF}
begin
{$IF CompilerVersion>31.0}
TestNumber := seTestNumber.Value;
sw := TStopwatch.StartNew;
Serializer := TJsonSerializer.Create;
try
Serializer.DateTimeZoneHandling := TJsonDateTimeZoneHandling.Utc;
for I := to TestNumber - do
begin
Foo := TObj1.Create;
try
Foo.field_s := 'Hello World';
Foo.field_i := ;
Foo.field_d := Now;
AJson := Serializer.Serialize<TObj1>(Foo);
finally
Foo.Free;
end; end;
Log('TJsonSerializer.Serialize:' + sw.ElapsedMilliseconds.ToString + ' ms ' + AJson); sw := TStopwatch.StartNew;
for I := to TestNumber - do
begin
Foo := Serializer.Deserialize<TObj1>(AJson);
try
finally
Foo.Free;
end;
end;
Log('TJsonSerializer.Deserialize:' + sw.ElapsedMilliseconds.ToString + ' ms'); finally
FreeAndNil(Serializer);
end;
Log('=======================');
{$ENDIF}
end; procedure TForm1.btnQJsonClick(Sender: TObject);
{$IFDEF QJson}
var
Foo: TObj1;
I: Integer;
sw: TStopwatch;
AJson: string;
Serializer: TQJson;
{$ENDIF}
begin
{$IFDEF QJson}
TestNumber := seTestNumber.Value;
sw := TStopwatch.StartNew;
Serializer := TQJson.Create;
for I := to TestNumber - do
begin
Foo := TObj1.Create;
try
Foo.field_s := 'Hello World';
Foo.field_i := ;
Foo.field_d := Now;
Serializer.FromRtti(Foo);
AJson := Serializer.AsJSON;
finally
Foo.Free;
end; end;
Log('TQJson.FromRtti:' + sw.ElapsedMilliseconds.ToString + ' ms ' + AJson); for I := to TestNumber - do
begin
Serializer.Parse(AJson);
Foo := TObj1.Create;
try
Serializer.ToRtti(Foo);
finally
Foo.Free;
end; end;
Serializer.Free;
Log('TQJson.ToRtti:' + sw.ElapsedMilliseconds.ToString + ' ms');
Log('=======================');
{$ENDIF}
end; procedure TForm1.FormCreate(Sender: TObject);
const
// D2010~D10.2
DelphiIDEVers: array [ .. ] of string = (
'Delphi 2010',
'Delphi XE',
'Delphi XE2',
'Delphi XE3',
'Delphi XE4',
'Delphi XE5',
'Delphi XE6',
'Delphi XE7',
'Delphi XE8',
'Delphi 10 Seattle',
'Delphi 10.1 Berlin',
'Delphi 10.2 Tokyo');
begin
{$IFDEF WIN64}
Caption := Caption + ' (64-bit)';
{$ENDIF}
Caption := Caption + ' - ' + DelphiIDEVers[Trunc(CompilerVersion)]; {$IF CompilerVersion<32.0}
btnJsonSerializer.Enabled := False;
{$ENDIF} {$IFNDEF XSuperObject}
btnXSuperObject.Enabled := False;
{$ENDIF}
{$IFNDEF JsonDataObjects}
btnJsonDataObjects.Enabled := False;
{$ENDIF}
{$IFNDEF QJson}
btnQJson.Enabled := False;
{$ENDIF}
ReportMemoryLeaksOnShutdown := True;
end; procedure TForm1.Log(const S: string);
begin
mmoLog.Lines.Add(S);
end; { TObj1 } constructor TObj1.Create;
begin
inherited;
{ SetLength(f_a, 4);
f_a[0] := '全能中间件';
f_a[1] := 'QQ:64445322';
f_a[2] := '淘宝:https://imaps.taobao.com/';
f_a[3] := '没有Unicode?\\//'; }
end; end.

Delphi 10.2 JSON与对象/结构体序列化性能提高100多倍的更多相关文章

  1. JSON还原为结构体

    JSON还原为结构体 1)JSON字符串还原为结构体: 2)访问结构体的字段值: 本例运行效果图: uses SynCommons; const // JSON字符串 JSON1 = '{' + #1 ...

  2. go语言之进阶篇json解析到结构体

    1.json解析到结构体 示例: package main import ( "encoding/json" "fmt" ) type IT struct { ...

  3. [GO]json解析到结构体

    package main import ( "encoding/json" "fmt" ) type IT struct { Company string `j ...

  4. Swift学习笔记(10):类和结构体

    目录: 基本 属性 方法 下标 继承 基本 使用class和struct关键字定义类和结构体. ・类是引用类型,结构体和枚举是值类型 ・值类型被赋予给一个变量.常量或被传递给一个函数时,已值拷贝方式传 ...

  5. go get请求 json字符串转为结构体

    package main import ( "io/ioutil" "fmt" "net/http" "encoding/json ...

  6. json数据转换成结构体

    package main import ( "encoding/json" "fmt" ) type IT1 struct { Company string ` ...

  7. 在线JSON转Go 结构体,在线JSON转Go Struct

    在线转换https://oktools.net/json2go

  8. 结构体序列为JSON

    结构体序列为JSON 本例运行效果图: uses SynCommons; const /// JSON字符串 JSON1 = '{' + #13#10 + '"glossary": ...

  9. 深入delphi编程理解之消息(二)发送消息函数及消息编号、消息结构体的理解

    一.delphi发送消息的函数主要有以下三个: (一).SendMessage函数,其原型如下: function SendMessage( hWnd: HWND; {目标句柄} Msg: UINT; ...

随机推荐

  1. tox 试用

    安装 pip install tox tox 使用 tox 包含一个tox.ini 文件,此文件可以自动,或者手工编写 tox.ini 文件格式 # content of: tox.ini , put ...

  2. HTML相对路径和绝对路径

     一.相对路径 相对路径不带有盘符,通常是以HTML网页文件为起点,通过层级关系描述目标图像的位置. 相对于文件自身出发,就是相对路径. 主要有以下几种情况:   ①文件和图片(html文档)在同一个 ...

  3. vue 把后端返回的图片和url链接生成的二维码用canvas 合成一张图片

    H5 页面在做某个活动的时候,有两种分享方式,一种是链接分享,一种是图片分享. 链接分享的话,如果是在微信里,就可引导用户利用微信浏览器自带的分享,根据sdk设置分享标题简介链接缩略图即可. 图片分享 ...

  4. [NOI2019]回家路线

    [NOI2019]回家路线 题目大意: 有\(n\)个站点,\(m\)趟车,每趟车在\(p_i\)时从\(x_i\)出发,\(q_i\)时到达\(y_i\). 若小猫共乘坐了\(k\)班列车,依次乘坐 ...

  5. 【转】Resource Localization in YARN

    一个Applciation运行在YARN上的流程为,从YARN Client向ResourceManager提交任务,将Applciation所需资源提交到HDFS中,然后ResourceManage ...

  6. Java 12 骚操作, switch居然还能这样玩!

    Java 13 都快要来了,12必须跟栈长学起! Java 13 即将发布,新特性必须抢先看! Java 12 中对 switch 的语法更友好了,建议大家看下栈长在Java技术栈微信公众号分享的&l ...

  7. 【软工实践】Beta冲刺(5/5)

    链接部分 队名:女生都队 组长博客: 博客链接 作业博客:博客链接 小组内容 恩泽(组长) 过去两天完成了哪些任务 描述 将数据分析以可视化形式展示出来 新增数据分析展示等功能API 服务器后端部署, ...

  8. Nginx访问路径添加密码保护

    创建口令文件 用openssl命令创建口令 openssl passwd -apr1 会产生一个hash口令, 然后和用户名一起, 以[用户名]:[hash口令]的格式写入文本文件即可 例如创建一个名 ...

  9. Laravel 入门常见问题汇总

    一.安装完成后想打开 Laravel 内置的登录页面,报错 解决方法: Laravel 利用 PHP5.4 的新特性 trait 内置了非常完善好用的简单用户登录注册功能,适合一些不需要复杂用户权限管 ...

  10. 爬虫中BeautifulSoup4解析器

    CSS 选择器:BeautifulSoup4 和 lxml 一样,Beautiful Soup 也是一个HTML/XML的解析器,主要的功能也是如何解析和提取 HTML/XML 数据. lxml 只会 ...