关于内存数据与 JSON

闲话:

正在用 WebBroker 写一个小网站,感受颇多:
1、如果是写一个小东西,应该先考虑下 WebBroker,因为它是分分钟即可上手的。
2、如果要写一个大东西,或许也应该考虑下 WebBroker,因为相比其他它有着最高的灵活度(但这只是想像,因为我只是写了一个小东西)。
3、Delphi 3 就有了 WebBroker,可惜现在才开始使用它;随着“服务器”与“网页程序”的普及,WebBroker 应该会被更多地提起。

期间用到了 json, 所以有了这篇博文。


Delphi 与内存数据的总结与回顾:

1、普通类型变量:储存一个数据。

2、数组:储存一组同类型的数据。

3、结构体:储存一组不同类型的数据;现在的结构体越来越复杂了,还可以操作这些个数据,越来越像“类”了。

4、“键值”对:我一般使用 TStringList,并常常用它代替 ini 文件类;同类还有 THashedStringList、TStringHash。

5、分类的 “键值”对:TMemIniFile、TIniFile; 另外还有人用 sqlite 做了一个可以储存二进制数据的 ini 文件类(忘了名了),我试过,非常好用(在官方示例中)。

6、内存数据表:现在应该首选 TFDMemTable 了(之前是 TClientDataSet)。

7、数据库级别的内存(或文件)数据:SqLite。

8、内存多叉树:JSON(或 XML);之前一直使用 SuperObject,以后会使用 System.JSON 更多些。


System.JSON?还是 SuperObject?

System.JSON 是官方的;这省去很多麻烦,这也是我考虑它的主要原因。

System.JSON “读写文件” 与 “格式化” 等功能,都有的功能也不如 SuperObject 方便。

System.JSON 主要考虑的是:从服务器生成 json,然后传输到客户端(特别是给 JavaScript 使用),不像 SuperObject 大而全。

如果不是太复杂的 json 应用,优先考虑 System.JSON 吧;如果需要更多功能,SuperObject 会更理想。


1 分钟了解 System.JSON:

其中的类不少,主要用到的有:TJSONObject、TJSONArray。


uses System.JSON;

//TJSONObject
procedure TForm1.Button1Click(Sender: TObject);
var
j: TJSONObject;
begin
j := TJSONObject.Create;
j.AddPair('aa', '111');
j.AddPair('bb', '222'); //输出字符串
Memo1.Text := j.ToString; //结果:{"aa":"111","bb":"222"}
{ToJSON 与 ToString 结果相同,但应尽量使用 ToString 而不是 ToJSON;因为 ToJSON 也是调用 ToString 并且还要重新分配内存} j.Free;
end; //TJSONArray
procedure TForm1.Button2Click(Sender: TObject);
var
a: TJSONArray;
begin
a := TJSONArray.Create;
a.Add(1); //TJSONArray 不像传统的数组,它可以接受几种不同类型的值;这应该算是对 Delphi 数组功能的扩充吧。
a.Add(2);
a.Add('aaa');
a.Add('bbb'); Memo1.Text := a.ToString; //结果:[1,2,"aaa","bbb"] a.Free;
end; //当 TJSONObject 嵌套其他对象时
procedure TForm1.Button3Click(Sender: TObject);
var
j: TJSONObject;
a: TJSONArray;
begin
j := TJSONObject.Create;
a := TJSONArray.Create('aaa', 'bbb'); //可以使用两个值初始化
a.Add(1);
a.Add(2); j.AddPair('arr', a); Memo1.Text := j.ToString; //结果:{"arr":["aaa","bbb",1,2]} // a.Free; {千万不要释放其内部对象,它的父对象会自动释放它}
j.Free;
end; //变通上例
procedure TForm1.Button4Click(Sender: TObject);
var
j: TJSONObject;
begin
j := TJSONObject.Create; j.AddPair('arr', TJSONArray.Create); with j.Values['arr'] as TJSONArray do
begin
Add('aaa');
Add('bbb');
Add(1);
Add(2);
end; Memo1.Text := j.ToString; //结果:{"arr":["aaa","bbb",1,2]}
j.Free;
end; //再变通
procedure TForm1.Button5Click(Sender: TObject);
var
j: TJSONObject;
a: TJSONArray;
begin
j := TJSONObject.Create; j.AddPair('arr', TJSONArray.Create); a := TJSONArray(j.Values['arr']);
a.Add('aaa');
a.Add('bbb');
a.Add(1);
a.Add(2); Memo1.Text := j.ToString; //结果:{"arr":["aaa","bbb",1,2]}
j.Free;
end; //添加其他类型的数
procedure TForm1.Button6Click(Sender: TObject);
var
j: TJSONObject;
a: TJSONArray;
begin
j := TJSONObject.Create;
j.AddPair('数值', TJSONNumber.Create(3.1415926));
j.AddPair('布尔真', TJSONTrue.Create);
j.AddPair('布尔假', TJSONFalse.Create);
j.AddPair('空值', TJSONNull.Create); Memo1.Text := j.ToString; //结果:{"数值":3.1415926,"布尔真":true,"布尔假":false,"空值":null}
j.Free;
end; //读取
procedure TForm1.Button7Click(Sender: TObject);
var
j: TJSONObject;
str: string;
num: Integer;
arr1,arr2: string;
begin
j := TJSONObject.Create;
j.AddPair('str', '111');
j.AddPair('num', TJSONNumber.Create(222));
j.AddPair('arr', TJSONArray.Create('arr1', 'arr2')); Memo1.Text := j.ToString; //结果:{"str":"111","num":222,"arr":["arr1","arr2"]} //JSON 是一组”键值对“,其中的”值“还有可能是”键值对“;
//所以 System.JSON 用 TJSONValue 来表示值类型,它是所有值类型的父类,当然也是 TJSONObject 的父类。 str := j.Values['str'].Value; //Values[] 调用的是 GetValue(), 更喜欢直接用后者
num := j.GetValue('num').Value.ToInteger; arr1 := TJSONArray(j.GetValue('arr')).Items[0].Value;
arr2 := TJSONArray(j.GetValue('arr')).Items[1].Value; Memo1.Lines.Add(Format('%s, %d, %s, %s', [str, num, arr1, arr2])); //111, 222, arr1, arr2 j.Free;
end; //遍历
procedure TForm1.Button8Click(Sender: TObject);
var
j: TJSONObject;
p: TJSONPair;
v: TJSONValue;
i: Integer;
a: TJSONArray;
begin
j := TJSONObject.Create;
j.AddPair('aa', '111');
j.AddPair('bb', '222');
j.AddPair('arr', TJSONArray.Create('arr1', 'arr2')); //集合遍历 TJSONObject
for p in j do
Memo1.Lines.Add(p.ToString); //"aa":"111" \\ "bb":"222" \\ "arr":["arr1","arr2"] //索引遍历 TJSONObject, 这里的 Get 函数已不提倡使用了
for i := 0 to j.Count - 1 do
Memo1.Lines.Add(j.Get(i).ToString); //"aa":"111" \\ "bb":"222" \\ "arr":["arr1","arr2"] //集合遍历 TJSONArray
for v in j.Values['arr'] as TJSONArray do
Memo1.Lines.Add(v.Value); //arr1 \\ arr2 //索引遍历 TJSONArray
a := j.Values['arr'] as TJSONArray;
for i := 0 to a.Count - 1 do
Memo1.Lines.Add(a.Items[i].Value); //arr1 \\ arr2
j.Free;
end; //存取到文件
procedure TForm1.Button9Click(Sender: TObject);
var
j1,j2: TJSONObject;
begin
//随便虚拟些数据
j1 := TJSONObject.Create(TJSONPair.Create('aaa', '111'));
j1.AddPair('bbb', j1.Clone as TJSONValue); ChDir('c:\temp'); //写入文件
with TStringList.Create do
begin
Add(j1.ToString);
SaveToFile('JsonTest.txt', TEncoding.UTF8); //结果:{"aaa":"111","bbb":{"aaa":"111"}}
Free;
end;
j1.Free; //从文件读取
with TStringList.Create do
begin
LoadFromFile('JsonTest.txt', TEncoding.UTF8);
j2 := TJSONObject.ParseJSONValue(Text) as TJSONObject;
Free;
end; Memo1.Text := j2.ToString; //结果:{"aaa":"111","bbb":{"aaa":"111"}}
j2.Free;
end; //修改、删除
procedure TForm1.Button10Click(Sender: TObject);
var
j: TJSONObject;
begin
j := TJSONObject.Create;
j.AddPair('aaa', '111');
j.AddPair('bbb', '222');
j.AddPair('ccc', '333'); //修改
j.Get('aaa').JsonValue := TJSONString.Create('11111'); //TJSONPair 的 JsonString、JsonValue 分别是 键、值
// j.Pairs[0].JsonValue := TJSONString.Create('11111'); //删除
j.RemovePair('bbb'); Memo1.Lines.Add(j.ToString); //结果:{"aaa":"11111","ccc":"333"} j.Free;
end; //根据路径添加数组的函数:
procedure CreateJSONPathArr(aJson: TJSONObject; const aPath: string; jArr: TJSONArray);
var
arr: TArray<string>;
path: string;
i: Integer;
begin
arr := aPath.Split(['.'], ExcludeEmpty);
for i := 0 to Length(arr) - 1 do
begin
if not aJson.TryGetValue(arr[i], aJson) then
begin
if i = Length(arr) - 1 then
aJson.AddPair(arr[i], jArr)
else begin
aJson.AddPair(arr[i], TJSONObject.Create);
aJson.TryGetValue(arr[i], aJson);
end;
end;
end;
end; //调用测试:
var
j: TJSONObject;
a: TJSONArray;
begin
j := TJSONObject.Create; a := TJSONArray.Create('11', '22');
CreateJSONPathArr(j, 'aaa.bbb.ccc', a); a := TJSONArray.Create('33', '44');
CreateJSONPathArr(j, 'aaa.bbb.ddd', a); j.Free;
end;

上一篇:Java基于OpenCV实现走迷宫(图片+路线展示)


下一篇:APT攻击将向云计算平台聚焦