用DELPHI的RTTI实现对象的XML持久化
去年我花了很多时间尝试用DELPHI进行基于XML的WEB应用开发。起初的设想是很美好的,但结果做出来的东西很简陋。一部分原因就在于XML到Object之间的数据绑定实现太麻烦(另一部分是因为对XSLT不熟,学习它花了很多时间)。
之前我一直是用DELPHI提供的XML Data binding来做的,基本做法是:先用工具(如XMLSPY)做好一个XML Schema(XSD),然后用XML Data binding生成DELPHI的接口和类。当然,一旦生成好就很方便了,在程序里我只要操作这个接口就好了,其中各个Field都会被变成属性,并且类型也都如我在XSD中的定义。但问题在于程序在开发过程中,总是会有一些变化的,在这种情况下,我就不得不同时开着XMLSPY修改XSD,然后重新用 XML Data binding的Wizard跑一遍,非常的麻烦。
所以当我想到数据集的对象化后,立即想到也可以用RTTI来实现Object的XML持久化--其实DELPHI6开始的SOAP实现就是用RTTI来实现Object到SOAP数据(就是XML)的转换的。显然我已经是非常的后知后觉了,当我在《强大的DELPHI RTTI--兼谈需要了解多种开发语言》一文中说到我的打算时,朋友Lex CHow回复我说他在大约一年前就做过了这方面的工作,我当即跟他要来了他的源码。lexlib是他写的是一个有很多功能的库,看上去结构有点像.net 的基本类库(当然没那么大^O^),Object的XML持久化只是其中的很小的一部分。因为我只需要这一部分,就没必要用这整个一个库这么麻烦,于是参考了lexlib并结合我在《用DELPHI的RTTI实现数据集的简单对象化》中已经实现的部分,做了一个简单的实现:
TMXMLPersistent = class(TObject)
public
class Procedure LoadObjFromXML( aNode : IXMLNode; aObj : TPersistent );
class Procedure SaveObjToXML( aNode : IXMLNode; aObj : TPersistent );
end; const
DefaultFilter : TTypeKinds = [tkInteger, tkChar, tkEnumeration,
tkFloat, tkString, tkSet, tkWChar, tkLString, tkWString, tkInt64]; { TMXMLPersistent } class procedure TMXMLPersistent.LoadObjFromXML(aNode: IXMLNode;
aObj: TPersistent);
Var
i : Integer;
pList : TMPropList;
pInfo : PPropInfo;
tmpObj: TObject;
begin
If ( aObj Is TMDataSetProxy ) Then
( aObj As TMDataSetProxy ).LoadFromXML( aNode )
Else
Begin
pList := TMPropList.Create( aObj );
Try
For i := 0 To pList.PropCount - 1 Do
Begin
pInfo := pList.Props[i];
If ( pInfo^.PropType^.Kind = tkClass ) Then
Begin
tmpObj := TObject( Integer( GetPropValue( aObj, pInfo^.Name ) ) );
If ( Assigned( tmpObj ) AND ( tmpObj Is TPersistent ) ) Then
LoadObjFromXML( aNode.ChildNodes[WideString(pInfo^.Name)],
tmpObj As TPersistent );
End
Else If ( pInfo^.PropType^.Kind In DefaultFilter ) Then
SetPropValue( aObj, pInfo^.Name,
String( aNode.ChildNodes[WideString( pInfo^.Name )].Text ) );
End;
Finally
pList.Free;
End;
End;
end; class procedure TMXMLPersistent.SaveObjToXML(aNode: IXMLNode;
aObj: TPersistent);
Var
i : Integer;
pList : TMPropList;
pInfo : PPropInfo;
tmpObj: TObject;
begin
If ( aObj Is TMDataSetProxy ) Then
( aObj As TMDataSetProxy ).SaveToXML( aNode )
Else
Begin
pList := TMPropList.Create( aObj );
Try
For i := 0 To pList.PropCount - 1 Do
Begin
pInfo := pList.Props[i];
If ( pInfo^.PropType^.Kind = tkClass ) Then
Begin
tmpObj := TObject( Integer( GetPropValue( aObj, pInfo^.Name ) ) );
If ( Assigned( tmpObj ) AND ( tmpObj Is TPersistent ) ) Then
SaveObjToXML( aNode.AddChild( WideString( pInfo^.Name ) ),
tmpObj As TPersistent );
End
Else If ( pInfo^.PropType^.Kind In DefaultFilter ) Then
aNode.AddChild( WideString( pInfo^.Name ) ).Text :=
GetPropValue( aObj, pInfo^.Name );
End;
Finally
pList.Free;
End;
End;
end;
这个实现应该说是很简单的。主要是三个部分(Load和Save的结构是相似的):
一是对TMDataSetProxy作特别处理,委托给这个类自己去处理它的实现,因为它与一般的类不同,需要通过ForEach遍历全部记录,这其实就是同时实现数据集的XML持久化。
二是对Class作递归处理,当然只支持从TPersistent派生的class。
三是一般的Field简单地转成String保存,其中借鉴了lexlib的Filter,只处理那些能简单地转成String的数据类型,过滤掉那些可能造成转换出错的类型。
上面的代码中用到的TMPropList见《用DELPHI的RTTI实现数据集的简单对象化》中的实现。
下面是用TMDataSetProxy实现的数据集的XML持久化。免去了需要通过TClientDataSet进行的麻烦,并且采用的是用Node记录字段的方式,.net也是采用这样的方式,与TClientDataSet所用的用Attribute记录字段的方式不同。虽然这样生成的 XML文件将会略大一些,但是好处也是显而易见的,特别是我是准备用在Web应用中的,用Node方式记录的XML,在用XSLT时会方便很多。
procedure TMDataSetProxy.LoadFromXML(aNode: IXMLNode);
Var
i, j : Integer;
pInfo : PPropInfo;
pRow : IXMLNode;
begin
For j := 0 To aNode.ChildNodes.Count - 1 Do
Begin
FDataSet.Append;
pRow := aNode.ChildNodes[j];
For i := 0 To FPropList.PropCount - 1 Do
Begin
pInfo := FPropList.Props[i];
If ( pInfo^.PropType^.Kind In DefaultFilter ) Then
SetVariant( i,
String( pRow.ChildNodes[WideString( pInfo^.Name )].Text ) );
End;
EndEdit;
End;
FDataSet.First;
end; procedure TMDataSetProxy.SaveToXML(aNode: IXMLNode);
Var
i : Integer;
pInfo : PPropInfo;
pRow : IXMLNode;
begin
While ForEach Do
Begin
pRow := aNode.AddChild( 'Row' );
For i := 0 To FPropList.PropCount - 1 Do
Begin
pInfo := FPropList.Props[i];
If ( pInfo^.PropType^.Kind In DefaultFilter ) Then
pRow.AddChild( WideString( pInfo^.Name ) ).Text
:= GetVariant( i );
End;
End;
end;
下面是一个简单的DEMO,其中包括了对数据集的XML持久化。注意Load的时候Employee成员连接的是ADODataSet2,它连接到一个包含了这几个字段的表,各字段类型与Employee表相同,但内容为空,并且去掉了EmployeeID的Identity。Load完成后,Employee表中这几个字段的内容将被复制到此表中。
TDemoCompany = class( TPersistent )
private
FEmployee : TDSPEmployee;
FCompany : String;
FCode : Integer;
published
property Employee : TDSPEmployee Read FEmployee Write FEmployee;
property Company : String Read FCompany Write FCompany;
Property Code : Integer Read FCode Write FCode;
End; procedure TForm1.SaveClick(Sender: TObject);
Var
demo : TDemoCompany;
begin
demo := TDemoCompany.Create;
demo.Employee := TDSPEmployee.Create( ADODataSet1 );
demo.Company := 'Demo company';
demo.Code := 987654;
Try
XMLDocument1.Active := true;
TMXMLPersistent.SaveObjToXML( XMLDocument1.AddChild( 'Demo' ), demo );
XMLDocument1.SaveToFile( 'temp.xml' );
XMLDocument1.Active := false;
Finally
demo.Employee.Free;
demo.Employee := Nil;
demo.Free;
End;
end; procedure TForm1.LoadClick(Sender: TObject);
Var
demo : TDemoCompany;
begin
demo := TDemoCompany.Create;
demo.Employee := TDSPEmployee.Create( ADODataSet2 );
Try
XMLDocument1.Active := true;
XMLDocument1.LoadFromFile( 'temp.xml' );
TMXMLPersistent.LoadObjFromXML( XMLDocument1.ChildNodes.Last, demo );
XMLDocument1.Active := false;
Edit1.Text := demo.Company;
Edit2.Text := IntToStr( demo.Code );
While ( demo.Employee.ForEach ) Do
With ListView1.Items.Add Do
Begin
Caption := IntToStr( demo.Employee.EmployeeID );
SubItems.Add( demo.Employee.FirstName );
SubItems.Add( demo.Employee.LastName );
SubItems.Add( FormatDateTime( 'yyyy-mm-dd', demo.Employee.BirthDate ) );
End;
Finally
demo.Employee.Free;
demo.Employee := Nil;
demo.Free;
End;
end;
终于可以告别那个麻烦的XML Data binding了,并且以后也不用写XSD了--虽然有好用的工具,但能省点事终归是好的。
http://blog.csdn.net/diligentcatrich/article/details/6934109
用DELPHI的RTTI实现对象的XML持久化的更多相关文章
- 用DELPHI的RTTI实现数据集的简单对象化
在<强大的DELPHI RTTI--兼谈需要了解多种开发语言>一文中,我说了一下我用DELPHI的RTTI实现了数据集的简单对象化.本文将详细介绍一下我的实现方法. 首先从一个简单 ...
- Delphi 的RTTI机制浅探3(超长,很不错)
转自:http://blog.sina.com.cn/s/blog_53d1e9210100uke4.html 目录========================================== ...
- 通过JAXB完成Java对象与XML之间的转换
Java对象转换XML的过程叫marshal. XML转换到Java对象的过程叫unmarshal. 一.Java对象转化为XML 这里省略getter和setter方法 通过标注@XMLRootEl ...
- java对象与XML相互转化
起因 最近在公司做了一次webservice相关的任务,其中我最敢兴趣的就是webservice接受到XML对应的流以后是如何方便的转化成java对象,而java对象又是如何生成对应的XML的. 目的 ...
- XmlSerializer 对象的Xml序列化和反序列化
http://www.cnblogs.com/yukaizhao/archive/2011/07/22/xml-serialization.html 这篇随笔对应的.Net命名空间是System.Xm ...
- 序列化对象为xml字符串
/// <summary> /// 序列化对象为xml字符串 /// </summary> /// <param name="obj" ...
- Java对象表示方式2:XStream实现对对象的XML化
上一篇文章讲到了使用Java原生的序列化的方式来表示一个对象.总结一下这种对象表示方式的优缺点: 1.纯粹的Java环境下这种方式可以很好地工作,因为它是Java自带的,也不需要第三方的Jar包的支持 ...
- JAVA对象和XML文档、原来他们之间还有这一出
最近项目开发中遇到一个问题,访问接口不再通过url地址请求的方式,而是 通过socket发送xml格式的报文到指定服务器来进行信息的统一认证.. 因此组装xml格式的报文字符串以及解析服务器返回的xm ...
- 玩转Java对象和XML相互转换
最近在项目中一直出现Java对象和XML之间的相互转换,一开始由于项目很庞大,我又是临时调度过去,导致在按照项目组长的要求进行写代码的同时,总是在这块云里雾里,最近才慢慢开始搞清楚项目中具体的使用缘由 ...
随机推荐
- 应用之间进行跳转,ComponentName的方式
从应用A跳转到应用B, 关键代码如下: 有以下几个注意点: 1.ComponentName cn = new ComponentName("com.terry", "co ...
- CURL基础
下载单个文件: #下载单个文件,默认将输出打印到标准输出中(STDOUT)中curl http://www.centos.org # 将文件下载到本地并命名为mygettext.html curl - ...
- php 学习笔记 数组1
1.一般情况下$name['tom']和$name[tom]是相同的:但没有引号的键不能和常量区别开,如:define('index', 5)时:$name['tom']和$name[tom]不同 2 ...
- PyQt中弹出对话框操作
经常有初学者搞不清楚如何在PyQt中弹出对话框,以及如何处理返回值.这篇文章会举例说明,界面采用手工编写. 我们一般说的对话框指的是模态对话框(Modal Dialogue Box),一旦弹出,就不能 ...
- Chapter 13 建造者模式
建造者模式又叫生成器模式:将一个产品的内部表象与产品的生成过程分割开来,从而可以使一个建造过程生成具有不同的内部表象的产品对象. 代码: package xiao; import java.util. ...
- oracle中if/else
oracle中if/else功能的实现的3种写法 1.标准sql规范 一.单个IF 1. if a=... then ......... end if; 2. if a=... then .... ...
- WEB开发:如何用js来模拟服务器的ajax响应,不依赖服务器来编写前端代码
一.问题的提出 目前web前端开发,主流的思路是: 1)编写静态的html文件(不使用模板技术,与服务器无关) 2)页面通过ajax与服务器交互,进行数据的传输,数据格式为json格式 这里存在一个问 ...
- c# 数据库编程(利用DataSet 和 DataAdaper对象操作数据库--跨表操作)
上篇文章我们介绍了如何利用DataSet 和 DataAdaper对象来对单张表进行操作. 本文我们将介绍如何进行跨表操作. 我们通过具体例子方式进行演示,例子涉及到三张表. 1)student表(学 ...
- TROUBLE SHOOTING: FRM-30425
关键字:Oracle Form buider FRM-30425 汇总 错误信息如下: FRM-30425: Summarized database item must reside in a blo ...
- c,assert 宏的实现
预备知识:#define _VAL(x) #x //#x的作用就是把x表达式变成一个字符串.(注意 : 不带换行符'\n' , 换行符ascii==10).如:_STR(i<100)printf ...