DELPHI 关于内存数据与 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;

DELPHI 关于内存数据与 JSON

上一篇:Professional C# 7 and .NET Core 2.0 - 简介


下一篇:Android:WebView(慕课网)