Delphi用QJSON解析JSON格式的数据

原本用superobject来解析JSON已经够用了,惋惜这个东东不能在移动端使用,因而找到QJSON来处理。json

这是一个国内高手写开源免费的东西,赞一个。数组

假入数据以下:编码

{"message":"ok","status":"1","state":"3","data":
[{"time":"2012-07-07 13:35:14","context":"客户已签收"},
 {"time":"2012-07-07 09:10:10","context":"离开 [北京石景山营业厅] 派送中,递送员
[温],电话[]"},
 {"time":"2012-07-06 19:46:38","context":"到达 [北京石景山营业厅]"},
 {"time":"2012-07-06 15:22:32","context":"离开 [北京石景山营业厅] 派送中,递送员
[温],电话[]"},
 {"time":"2012-07-06 15:05:00","context":"到达 [北京石景山营业厅]"},
 {"time":"2012-07-06 13:37:52","context":"离开 [北京_同城中转站] 发往 [北京石景山
营业厅]"},
 {"time":"2012-07-06 12:54:41","context":"到达 [北京_同城中转站]"},
 {"time":"2012-07-06 11:11:03","context":"离开 [北京运转中心驻站班组] 发往 [北京_
同城中转站]"},
 {"time":"2012-07-06 10:43:21","context":"到达 [北京运转中心驻站班组]"},
 {"time":"2012-07-05 21:18:53","context":"离开 [福建_厦门支公司] 发往 [北京运转中
心_航空]"},
 {"time":"2012-07-05 20:07:27","context":"已取件,到达 [福建_厦门支公司]"}
]} 

用QJSON解析以下:spa

procedure TForm15.Button1Click(Sender: TObject);
var
  aqjson,aqjsonarr : TQJSON;
  i : Integer;
  stime, scontext : string;
begin
  aqjson := TQJSON.Create;
  aqjson.parse(memo1.lines.text);
  if aqjson.ValueByName('message', '') = 'ok' then
  begin
    memo2.Clear;
    aqjsonarr := aqjson.ItemByName('data');
    for i := 0 to aqjsonarr.Count - 1 do
    begin
      stime := aqjsonarr.Items[i].ValueByName('time', '');
      scontext := aqjsonarr.Items[i].ValueByName('context', '');

      Memo2.Lines.Add(stime+'----'+scontext);
    end;
  end;
end;

能够看到QJSON的解析仍是很方便的。code

不过这种格式存在大量冗余数据——每一个数据项都携带了字段信息,其实能够只返回一次字段信息便可。orm

数据精简以下:blog

{"message":"ok","status":"1","state":"3","data":
["2012-07-07 13:35:14","客户已签收",
 "2012-07-07 09:10:10","离开 [北京石景山营业厅] 派送中,递送员[温],电话[]",
]} 

能够看到数组里面的串再也不是JSON格式(Key:Value)的了,这时不能再使用ValueByName,而直接使用Value。图片

procedure TForm15.Button2Click(Sender: TObject);
var
  aqjson,aqjsonarr : TQJSON;
  i : Integer;
  stime, scontext : string;
begin
  aqjson := TQJSON.Create;
  aqjson.parse(memo3.lines.text);
  if aqjson.ValueByName('message', '') = 'ok' then
  begin
    memo2.Clear;
    aqjsonarr := aqjson.ItemByName('data');
    for i := 0 to aqjsonarr.Count - 1 do
    begin
      stime := aqjsonarr.Items[i].ValueByName('time', '');
      scontext := aqjsonarr.Items[i].ToString;

      Memo2.Lines.Add(stime+'----'+scontext);
    end;
  end;
end;

实际编码中,会存在返回图片到客户端的状况,若是也采用JSON格式传输的话,须要把图片转成Base64格式的传包装,而后再传输到客户端解析。内存

这里是一个演示,首先把图片转成流:字符串

Image1.Picture.Graphic.SaveToStream(ss);

而后编码成base64格式的:

EncodeStream(ss, ss1);

注意ss和ss1的定义:

var
  ss: TMemoryStream;
  ss1,ss2 : TStringStream;
EncodeStream的调用须要引用EncdDecd.pas单元。

而后把流转成字符串
var
  sdata : string;
begin
...
  sData := ss1.DataString;
...
end;

再把该字符串包装到JSON串:

var
  aqjson : TQJSON;
begin
  aqjson := TQJSON.Create;
  aqjson.Parse(memo3.Lines.Text);
...
  aqjson.AddArray('pic').Add.AsString :=sdata;
...
end;

这样图片就打包到JSON里面了,传到客户端之后,再反过来解析便可:

ss2 := TStringStream.Create(aqjson.ItemByName('pic').Items[0].value);
DecodeStream(ss2,ss);//将base64字符流还原为内存流
ss.Position := 0; // 必须
Image2.Picture.Graphic.LoadFromStream(ss);
相关文章
相关标签/搜索