本文将结合
面向对象设计原则实践来介绍如何将一个Java对象树输出到XML文件模块。
需求
项目功能:查询数据并生成XML文件然后上传至指定服务器
本模块功能:实现Java对象树输出到 XML文件。
要求
1. 支持对象及属性的扩展,而XML输出模块代码基本不变。
2. 考虑到内存压力,要求该模块实现以追加方式操作XML文件。
输入
Java对象树(提供一个类以方法,作为该对象的操作入口)
输出
XML文档
数值传输对象( DTO,Data Transfer Object),仅包含属性和setter/getter方法。如例1所示。属性的型别基本固定(例如String, int,boolean,List等),对于List的属性,可能包含另一个Java Object(如例3所示)。XML输出格式很简单,除去头部信息之外基本是attribute+value。例2是例1对象XML输出片段。
例1
例2
例3
分析
先介绍一下XML输出模块的旧版本。它直接处理判断对象的类型,并显示获取其getter方法,得到属性的值后操作XML文档(即dom4j的Document)。如下所示 :
设计
对象解析
如说封装对象及属性的变化。数值对象有属性组成(以及setter/getter方法,此处忽略);Property(及属性)由类型(已封装成Type类,它将常用的类型定义成类并将静态实例暴露给外界),名称和Alias组成。对象树实际上是数据结构中的树。属性就是叶子。List型属性是可解析的,因此有字节点;非List型属性是末级节点。是否可解析可通过调用listable方法判断。
Property类实现了属性初始化(加载所有属性并生成相应对象供解析时使用)。利用反射机制可获得每个属性的的getter方法并在对象上调用该方法以取得其值。遍历所有节点可构造成一棵Element(稍后还会讲到该接口)树。
XML输出
dom4j是一个开源的XML工具,它支持XML解析和输出。但输出XML的过程采用内存构造Document方式。输出时调用Document.write方法写文件。前面已经提到,由于对象树非常庞大,一次构造完整的Document对象将导致OutOfMemoryError错误。如果能够实现以Append操作XML文件方式替换前面提到的方式,将会避免该问题出现。
Element接口定义了XML元素这个概念。它支持设置元素名称,文本和IO输出操作。XMLStream实现了Element接口,可以表示一个XMl元素并能输出到指定设备。XMLWriter实现了对Element输出到XML的过程。
需求
项目功能:查询数据并生成XML文件然后上传至指定服务器
本模块功能:实现Java对象树输出到 XML文件。
要求
1. 支持对象及属性的扩展,而XML输出模块代码基本不变。
2. 考虑到内存压力,要求该模块实现以追加方式操作XML文件。
输入
Java对象树(提供一个类以方法,作为该对象的操作入口)
输出
XML文档
数值传输对象( DTO,Data Transfer Object),仅包含属性和setter/getter方法。如例1所示。属性的型别基本固定(例如String, int,boolean,List等),对于List的属性,可能包含另一个Java Object(如例3所示)。XML输出格式很简单,除去头部信息之外基本是attribute+value。例2是例1对象XML输出片段。
例1
java 代码
- public class FromTo{
- private String fromNo;
- private String toNo;
- private FromTo() { }
- public void setFromNo(String fromNo){
- this.fromNo = fromNo;
- }
- public String getFromNo() {
- return fromNo;
- }
- public void setToNo(String toNo){
- this.toNo = toNo;
- }
- public String getToNo() {
- return toNo;
- }
- }
例2
xml 代码
- <fromNo>DLS</fromNo>
- <toNo>BJS</toNo>
例3
java 代码
- public class CarrierFlight implements Serializable {
- /**
- *
- */
- private static final long serialVersionUID = 4295207854877905411L;
- private List apFlightNoList = new ArrayList(); //apFlightNoList存储FromTo类型的对象
- private List exFlightNoList = new ArrayList(); //同上
- private String carrierCode;
- private String rbd;
- public CarrierFlight(){ }
- /**
- * @param apFlightNoList
- */
- public void setApFlightNoList(List apFlightNoList) {
- this.apFlightNoList = apFlightNoList;
- }
- /**
- * @return
- */
- public List getApFlightNoList() {
- return apFlightNoList;
- }
- /**
- * @param apFlightNos
- */
- public void setApFlightNoList(ApFlightNo[] apFlightNos){
- if(!ArraysUtil.foreachable(apFlightNos))
- return;
- for(int i=0;i < apFlightNos.length;i++)
- this.apFlightNoList.add(apFlightNos[i];
- }
- /**
- * @param exFlightNoList
- */
- public void setExFlightNoList(List exFlightNoList) {
- this.exFlightNoList = exFlightNoList;
- }
- /**
- * @return
- */
- public List getExFlightNoList() {
- return exFlightNoList;
- }
- /**
- * @param exFlightNos
- */
- public void setExFlightNoList(ExFlightNo[] exFlightNos){
- if(!ArraysUtil.foreachable(exFlightNos))
- return;
- for(int i=0;i < exFlightNos.length;i++)
- this.exFlightNoList.add(exFlightNos[i];
- }
- /**
- * @param carrierCode
- */
- public void setCarrierCode(String carrierCode){
- this.carrierCode = carrierCode;
- }
- /**
- * @return
- */
- public String getCarrierCode(){
- return this.carrierCode;
- }
- /**
- * @param rbd
- */
- public void setRBD(String rbd){
- this.rbd = rbd;
- }
- /**
- * @return
- */
- public String getRBD(){
- return this.rbd;
- }
- }
分析
先介绍一下XML输出模块的旧版本。它直接处理判断对象的类型,并显示获取其getter方法,得到属性的值后操作XML文档(即dom4j的Document)。如下所示 :
java 代码
- Object obj = getFromToObject();
- if(fromTo instanceof FromTo){
- FromTo fromto = (FromTo)obj;
- String fromNo = fromto.getFromNo();
- String toNo = fromto.getToNo();
- if(fromNo!=null && fromNo.length()>0){
- xml.output("<fromNo>"+fromNo+"</fromNo>");
- }
- if(toNo!=null && toNo.length()>0){
- xml.output("<toNo>"+toNo+"</toNo>");
- }
- }
- //....
面向对象设计两个重要原则:单一职责原则(SRP)和开放-闭合原则(OCP)。SRP要求对每个类仅有一个引起它变化的原因。OCP要求对扩展开放,对修改封闭。<o:p></o:p>
上述代码很容易理解,但它充满臭味:不必要的重复和脆弱性(想象如何把例3展示的CarrierFlight输出到XML)。引起上述代码发生变化的原因很多:Java对象树中某个对象的属性或者XML输出格式改变。同时,缺乏抽象导致该类很难扩展(例如增加对新类型的Object的支持而代码不改变),每次修改都必须改动同一个类。<o:p></o:p>
<o:p>遵循SRP和OCP,需要:</o:p>
<o:p>1.把XML输出和对象解析过程分离</o:p>
<o:p>2.把频繁变化的部分抽象
</o:p>
设计
对象解析
如说封装对象及属性的变化。数值对象有属性组成(以及setter/getter方法,此处忽略);Property(及属性)由类型(已封装成Type类,它将常用的类型定义成类并将静态实例暴露给外界),名称和Alias组成。对象树实际上是数据结构中的树。属性就是叶子。List型属性是可解析的,因此有字节点;非List型属性是末级节点。是否可解析可通过调用listable方法判断。
Property类实现了属性初始化(加载所有属性并生成相应对象供解析时使用)。利用反射机制可获得每个属性的的getter方法并在对象上调用该方法以取得其值。遍历所有节点可构造成一棵Element(稍后还会讲到该接口)树。
XML输出
dom4j是一个开源的XML工具,它支持XML解析和输出。但输出XML的过程采用内存构造Document方式。输出时调用Document.write方法写文件。前面已经提到,由于对象树非常庞大,一次构造完整的Document对象将导致OutOfMemoryError错误。如果能够实现以Append操作XML文件方式替换前面提到的方式,将会避免该问题出现。
Element接口定义了XML元素这个概念。它支持设置元素名称,文本和IO输出操作。XMLStream实现了Element接口,可以表示一个XMl元素并能输出到指定设备。XMLWriter实现了对Element输出到XML的过程。