在delphi中,数据集是最常用数据存取方式。因此,必须建立JSON与TDataSet之间的互转关系,实现数据之间通讯与转换。值得注意的是,这只是普通的TDataset与JSON之间转换,由于CDS包含了Delta数据包,其数据格式远比普通的TDataset更复杂。

数据集字段信息,是一个完整的字典信息。因此,我们在JSON必须也建立字典信息,才能创建数据集的字段信息。我们设置其JSON信息如下:

COLS:[字段列表信息],如:

"Cols":[{"JsonType":"integer","FieldIndex":0,"FieldType":"Integer","FieldSize":0,"FieldName":"ID","Required":false},{"JsonType":"string","FieldIndex":1,"FieldType":"String","FieldSize":100,"FieldName":"Title","Required":false},{"JsonType":"variant","FieldIndex":2,"FieldType":"Blob","FieldSize":0,"FieldName":"Picture","Required":false}]

数据信息以Data做节点,也是一个数组嵌套记录信息:

"Data":[记录集信息]

废话少说,直接上代码:unit uDBJson;

interface

uses

SysUtils,Classes,Variants,DB,DBClient,SuperObject;

type

TTableJSon = class

private

const cstFieldType = 'FieldType';

const cstFieldName = 'FieldName';

const cstFieldSize = 'FieldSize';

const cstJsonType = 'JsonType';

const cstRequired = 'Required';

const cstFieldIndex = 'FieldIndex';

const cstCols= 'Cols';

const cstData= 'Data';

public

class function JSonFromDataSet(DataSet:TDataSet):string;

class function CreateFieldByJson(Fields:TFieldDefs;ColsJson:ISuperObject):Boolean;

class function ImportDataFromJSon(DataSet:TDataSet;DataJson:ISuperObject):Integer;

class function CDSFromJSon(CDS:TClientDataSet;Json:ISuperObject):Boolean;

class function GetValue(Json:ISuperObject;const Name:string):Variant;

class function CreateJsonValue(Json:ISuperObject;const Name:string;const Value:Variant):Boolean;

class function CreateJsonValueByField(Json:ISuperObject;Field:TField):Boolean;

class function GetValue2Field(Field:TField;JsonValue:ISuperObject):Variant;

end;

implementation

uses TypInfo,encddecd;

{ TTableJSon }

class function TTableJSon.CDSFromJSon(CDS: TClientDataSet;

Json: ISuperObject): Boolean;

var

ColsJson:ISuperObject;

begin

Result := False;

if Json = nil then

Exit;

CDS.Close;

CDS.Data := Null;

//创建字段

ColsJson := Json.O[cstCols];

CreateFieldByJson(CDS.FieldDefs,ColsJson);

if CDS.FieldDefs.Count >0 then

CDS.CreateDataSet;

ImportDataFromJSon(CDS,Json.O[cstData]);

Result := True;

end;

class function TTableJSon.CreateFieldByJson(Fields: TFieldDefs;

ColsJson: ISuperObject): Boolean;

var

SubJson:ISuperObject;

ft:TFieldType;

begin

Result := False;

Fields.DataSet.Close;

Fields.Clear;

for SubJson in ColsJson do

begin

ft := TFieldType(GetEnumValue(TypeInfo(TFieldType),'ft'+SubJson.S[cstFieldType]));

if ft= ftAutoInc then //自增字段不能录入,必须更改

ft := ftInteger;

Fields.Add(SubJson.S[cstFieldName],ft,SubJson.I[cstFieldSize],SubJson.B[cstRequired]);

end;

Result := True;

end;

class function TTableJSon.CreateJsonValue(Json: ISuperObject;

const Name: string; const Value: Variant): Boolean;

begin

Result := False;

Json.O[Name] := SO(Value);

Result := True;

end;

class function TTableJSon.CreateJsonValueByField(Json: ISuperObject;

Field: TField): Boolean;

begin

Result := False;

if Field Is TDateTimeField then

Json.O[Field.FieldName] := SO(Field.AsDateTime)

else if Field is TBlobField then

Json.S[Field.FieldName] := EncodeString(Field.AsString)

else

Json.O[Field.FieldName] := SO(Field.Value);

Result := True;

end;

class function TTableJSon.GetValue(

Json: ISuperObject;const Name: string): Variant;

begin

case Json.DataType of

stNull: Result := Null;

stBoolean: Result := Json.B[Name];

stDouble: Result := Json.D[Name];

stCurrency: Result := Json.C[Name];

stInt: Result := Json.I[Name];

stString: Result := Json.S[Name];

end;

end;

class function TTableJSon.GetValue2Field(Field: TField; JsonValue:ISuperObject): Variant;

begin

if JsonValue.DataType = stNull then

Result := Null

else if Field is TDateTimeField then

Result := JavaToDelphiDateTime(JsonValue.AsInteger)

else if (Field is TIntegerField) or (Field is TLargeintField) then

Result := JsonValue.AsInteger

else if Field is TNumericField then

Result := JsonValue.AsDouble

else if Field is TBooleanField then

Result := JsonValue.AsBoolean

else if Field is TStringField then

Result := JsonValue.AsString

else if Field is TBlobField then

Result := DecodeString(JsonValue.AsString)

end;

class function TTableJSon.ImportDataFromJSon(DataSet: TDataSet;

DataJson: ISuperObject): Integer;

var

SubJson:ISuperObject;

i:Integer;

iter: TSuperObjectIter;

begin

if not DataSet.Active then

DataSet.Open;

DataSet.DisableControls;

try

for SubJson in DataJson do

begin

DataSet.Append;

if ObjectFindFirst(SubJson,iter) then

begin

repeat

if DataSet.FindField(iter.Ite.Current.Name)<>nil then

DataSet.FindField(iter.Ite.Current.Name).Value :=

GetValue2Field(

DataSet.FindField(iter.Ite.Current.Name),

iter.Ite.Current.Value);

until not ObjectFindNext(iter) ;

end;

DataSet.Post;

end;

finally

DataSet.EnableControls;

end;

end;

class function TTableJSon.JSonFromDataSet(DataSet:TDataSet):string;

procedure GetFieldTypeInfo(Field:TField;var Fieldtyp,JsonTyp:string);

begin

Fieldtyp := GetEnumName(TypeInfo(tfieldtype),ord(Field.DataType));

Delete(Fieldtyp,1,2);

if Field is TStringField then

JsonTyp := 'string'

else if Field is TDateTimeField then

JsonTyp := 'integer'

else if (Field is TIntegerField) or (Field is TLargeintField) then

JsonTyp := 'integer'

else if Field is TCurrencyField then

JsonTyp := 'currency'

else if Field is TNumericField then

JsonTyp := 'double'

else if Field is TBooleanField then

JsonTyp := 'boolean'

else

JsonTyp := 'variant';

end;

var

sj,aj,sj2:ISuperObject;

i:Integer;

Fieldtyp,JsonTyp:string;

List:TStringList;

begin

sj := SO();

//创建列

aj := SA([]);

List := TStringList.Create;

try

List.Sorted := True;

for i := 0 to DataSet.FieldCount - 1 do

begin

sj2 := SO();

GetFieldTypeInfo(DataSet.Fields[i],Fieldtyp,JsonTyp);

sj2.S[cstFieldName] := DataSet.Fields[i].FieldName;

sj2.S[cstFieldType] := Fieldtyp;

sj2.S[cstJsonType] := JsonTyp;

sj2.I[cstFieldSize] := DataSet.Fields[i].Size;

sj2.B[cstRequired] := DataSet.Fields[i].Required;

sj2.I[cstFieldIndex] := DataSet.Fields[i].Index;

aj.AsArray.Add(sj2);

List.Add(DataSet.Fields[i].FieldName+'='+JsonTyp);

end;

sj.O['Cols'] := aj;

//创建数据集的数据

DataSet.DisableControls;

DataSet.First;

aj := SA([]);

while not DataSet.Eof do

begin

sj2 := SO();

for i := 0 to DataSet.FieldCount - 1 do

begin

//sj2.S[IntToStr(DataSet.Fields[i].Index)] := VarToStrDef(DataSet.Fields[i].Value,'');

if VarIsNull(DataSet.Fields[i].Value) then

sj2.O[DataSet.Fields[i].FieldName] := SO(Null)

else

begin

CreateJsonValueByField(sj2,DataSet.Fields[i]);

end;

end;

aj.AsArray.Add(sj2);

DataSet.Next;

end;

sj.O['Data'] := aj;

Result := sj.AsString;

finally

List.Free;

DataSet.EnableControls;

end;

end;

end.

调用示例:

//数据集转JSON对象或JSON文本

var

json:TTableJSon;

s:string;

begin

S := json.JSonFromDataSet(ADODataSet1);

//在用TStringStream读入字符串S,存成文本,看看其格式.

end;

//JSON对象或文本,装载到数据集

var

json:ISuperObject;

begin

json := TSuperObject.ParseFile('json.txt',False);

TTableJSon.CDSFromJSon(cdsJSON,json);

end;

http://blog.csdn.net/diligentcatrich/article/details/6916288

Delphi中JSon SuperObject 使用:数据集与JSON对象互转的更多相关文章

  1. SUPEROBJECT序列数据集为JSON

    // SUPEROBJECT 序列数据集 cxg 2017-1-12// {"data":[{"c1":1,"c2":1}]};// DEL ...

  2. delphi 中TStringList Clear 方法的时候该对象有没有被释放

    delphi 中TStringList 通过function AddObject(const S: string; AObject: TObject): Integer; 方法添加了一个对象,请问我在 ...

  3. Android开发中的Json字符串与复杂的嵌套对象互转。

    Gson 可能是大家都觉得比较简单吧.我发现用JSONObject和网上下载的JSONHelper类使用起来很无语,只能解析简单的单层对象,如果有嵌套的就不能直转转成可用对象了.所以网上找了一会儿,发 ...

  4. DELPHI中四种EXCEL访问技术实现

    一.引言 EXCEL在处理中文报表时功能非常强大,EXCEL报表访问也是信息系统开发中的一个重要内容,本文总结以往开发中所用到的几中EXCEL文件访问方法,在实际工作中也得到了很好的验证,本文列举了其 ...

  5. Delphi中比较两个对象是否一致及地址是否相同[转]

    在delphi中,C#也是如此,对象的地址与对象变量(引用)的地址不是同一个概念.要加以区别. procedure TForm1.btn1Click(Sender: TObject); var    ...

  6. Delphi中的Free和Nil和freeandnil函数

    Delphi中的Free和Nil 在Delphi中释放对象资源时一般用Obj.Free(Obj为一个实例名),不过程Delphi中还有一个FreeAndNil(对象名)函数,那么用哪个好呢?Free和 ...

  7. Delphi中使用ISuperObject解析Json数据

    Java.Php等语言中都有成熟的框架来解析Json数据,可以让我们使用很少的代码就把格式化好的json数据转换成程序可识别的对象或者属性,同时delphi中也有这样的组件来实现此功能,即Isuper ...

  8. Delphi中Json格式读写

    Json是一种轻量级传输数据格式,广泛应用互联网和各应用中.json主要採用键值对来表示数据项.多个数据项之间用逗号分隔,也能够用于数组.以下注重介绍一下在delphi中使用json,在delphi中 ...

  9. Delphi中DataSet和JSON的互转

    //1)数据集转换为JSON字符串://需USES System.JSON; function DataSetToJson(ADataset: TDataSet): string; // [{&quo ...

随机推荐

  1. PHP 自学之路-----XML编程(Xpath技术,simpleXml技术)基础入门

    XPAth技术 XPath的设计的核心思想,可以通过xpath迅速简介的定位到你希望查找的节点.主要目的是描述节点相对其他节点的位置,可以取得所有符合条件的节点,成为[位置路径]. Xapth主要用来 ...

  2. 正确的 zip 压缩与解压代码

    网上流传的zip压缩与解压 的代码有非常大的问题 尽管使用了ant进行压缩与解压,可是任务的流程还是用的java.util.zip 的方式写的,我在使用的过程中遇到了压缩的文件夹结构有误,甚至出现不同 ...

  3. Python用subprocess的Popen来调用系统命令

    当我们须要调用系统的命令的时候,最先考虑的os模块.用os.system()和os.popen()来进行操作.可是这两个命令过于简单,不能完毕一些复杂的操作,如给执行的命令提供输入或者读取命令的输出, ...

  4. CPU指令的流水线运行

    指令集是CPU体系架构的重要组成部分.C语言的语法是对解决现实问题的运算和流程的方法的高度概况和抽象,其主要为算术.逻辑运算和分支控制,而指令集就是对这些抽象的详细支持,汇编仅仅只是是为了让开发者更好 ...

  5. Xcode的Architectures、Valid Architectures和Build Active Architecture Only属性

    Architectures 这代表,在这个项目里你想要Xcode编译的目标设备列表. Valid Architectures 还不是太明确这个设置的意图,但是一般来说是不需要更改的,和Architec ...

  6. uboot: 理解uboot要看哪些书

    概览:

  7. CentOS安装rar及用法

    1.下载安装rar wget http://www.rarsoft.com/rar/rarlinux-x64-5.4.b3.tar.gztar -zxvf rarlinux-x64-.tar.gz - ...

  8. USACO Hamming Codes DFS 构造

    我还是用了很朴素的暴力匹配A了这题,不得不感叹USACO时间放的好宽... /* ID: wushuai2 PROG: hamming LANG: C++ */ //#pragma comment(l ...

  9. JSON数组分配输出每个li

    有这么一个JSON数组,需求是只需要输出每个数组里面的某个值,不需要全部输出来. var data = [ { ", "Cost":"13,642.41&quo ...

  10. php下载远程图片方法总结(curl手动解析header)curl跳转问题解决

    常用方法一般有:. file_get_contents file_put_contents readfile($file) //效率很高. 一般代码: /** * 抓取远程图片 * * @param ...