XML和XSLT实现代码生成器(IV)

类别:Java 点击:0 评论:0 推荐:
XML和XSLT实现代码生成器(IV) 结果处理

       本文第一部分描述了如何使用静态XML文档和XSLT以及一个简单的Java转换程序实现基本的代码生成器,然而通过分析结果,我的实现至少还有两点是十分原始的,首先代码输出结果的格式非常不理想(参看图2.1),输出代码之间完全没有的空行、缩进,导致代码非常难以阅读,这就需要通过一个程序将原始结果过滤为符合阅读需求的Java源代码,可以将这一步称为代码美容。一种可选的方法是使用现存的产品自动清理Java代码,例如JIndent (http://www.jindent.com);另一种方法是自己动手美容代码,列表2.2显示了我的简单实现

Pic 2.1 Result of Generated Code(Using IE Browser)

package com.xs.xgen.util;

import java.io.*;

import java.util.*;

/**

 * <p>Title: Code Generator based on XML and XSLT</p>

 * <p>Description: Beta Version For Code Generator</p>

 * <p>Copyright: xchu@Copyright (c) 2004</p>

 * <p>University: Melbourne University</p>

 * @author Xingchen Chu

 * @version 0.1

 */

public class IndentUtil {

  /**

   * <description> add indent to the content of input file output </description>

   * @param java.io.File input

   * @param java.io.File output

   */

  public static void indentJavaSource(File input, File output){

try{

      //read all the content at one time and write them to the string buffer

      BufferedReader in = new BufferedReader(new FileReader(input));

      StringBuffer sb=new StringBuffer();

      String line=null;

      while((line=in.readLine())!=null){

        sb.append(line);

      }

      String content = sb.toString();

      FileWriter writer=new FileWriter(output);

indentContent(writer,content,0,0); //indent from the beginning of the content

    }catch(Exception e){

      throw new RuntimeException(e);

}finally{

      writer.close();

}

  }

  private static void indentContent(Writer writer,String content,

int begin,int indent)throws IOException{

    if(begin>content.length()-1){ //now the position is the end of the content

      writer.flush();

      return;

    }

    char currentChar = content.charAt(begin); //get the current char of the content

    if(currentChar=='}'){

      writer.write("\r\n");

      indent-=5;

      for(int j=0;j<indent;j++){

        writer.write(" ");

      }

writer.write(currentChar);

//check whether is the end of the class file

      if((begin+1)<content.length()&&content.charAt(begin+1)!=')'){

        writer.write("\r\n");

        for (int j = 0; j < indent; j++) {

          writer.write(" ");

        }

      }

    }else{

      writer.write(currentChar);

      if(currentChar=='{'){

        writer.write("\r\n");

        indent+=5;

        for(int j=0;j<indent;j++){

          writer.write(" ");

        }

}else if(currentChar==';'){

  //check whether is the end of the method

        if((begin+1)<content.length()&&content.charAt(begin+1)!='}'){

          writer.write("\r\n");

          for (int j = 0; j < indent; j++) {

            writer.write(" ");

          }

        }

      }else {///nothing to do}

    }

    indentContent(writer,content,begin+1,indent);//recursively evaluate next char

  }

}

list 2.2 IndentUtil.java(recursive version of indent method)

动态生成XML

       除了之前提到的代码美容外,我的初试方案中的另外一个缺陷是XML文档是静态写入的,考虑到图形化代码生成器的需要,静态方式将无法满足图形化的要求。因此,这里必须实现XML文档的动态生成功能。可以采取SAX或着DOM标准实现XML动态生成,在此我利用JDOM API,因为它是DOM的面向对象版本,比直接使用DOM更加容易。然后利用JDOM的方法将XML JDOM节点转换为标准的DOM节点,通过JAXP的Transformer对象实现从DOM节点并利用XSLT到Java代码的转换。

                                   Pic 2.3 UML for My Data Model and JDOM Util

数据模型

       这里我将采用如下策略生成数据模型

1.首先根据之前定义的DTD定制数据结构并采取如下方式,DTD中的任何ELEMENT如果其内容是#PCDATA且没有任何属性,则使用String,否则定义对应的Java类。

2.根据DTD的ELEMENT定义中的通配符,如果是*或+则在对应元素的Java类中使用Collection表示其子元素。

3.对于任何属性列表,在Java类定义中使用Map对象表示。

例如对于Property元素,它的定义为<!ELEMENT property (name,exception*)>,

同时还为其定义了属性列

<!ATTLIST property

       type CDATA #REQUIRED

       access (public | protected | private | package) #REQUIRED

       set (yes | no) #REQUIRED

       get (yes | no) #REQUIRED

>

根据我们的规则,其Java类定义如下

package com.xs.xgen.javabean;

import java.util.*;

public class PropertyData {

  private Map attributes = new HashMap();

  private String name;

  private Collection exceptions;

  public PropertyData(String name,Collection exceptions,Map attributes) {

    this.name=name;

    this.exceptions=exceptions;

    this.attributes=attributes;

  }

  public String getName(){

    return name;

  }

  public Map getAttributes(){

    return Collections.unmodifiableMap(attributes);

  }

  public Collection getExceptions(){

    return Collections.unmodifiableCollection(exceptions);

  }

}

list 2.4 PropertyData.java

同理,JavaBean元素和Package元素的定义就十分直观了

package com.xs.xgen.javabean;

import java.util.*;

public class JavaBeanData {

  private String name;

  private PackageData packageData ;

  private Collection implement;

  private Collection propertyData;

 

  public JavaBeanData(String name,PackageData packageData,Collection implement,Collection propertyData) {

    this.name=name;

    this.packageData=packageData;

    this.implement=implement;

    this.propertyData=propertyData;

  }

  public String getName(){ return name; }

  public PackageData getPackageData(){ return packageData;}

  public Collection getImplement(){

    return Collections.unmodifiableCollection(implement);

  }

  public Collection getPropertyData(){

    return Collections.unmodifiableCollection(propertyData);

  }

}

                        list 2.5 JavaBeanData.java

package com.xs.xgen.javabean;

public class PackageData {

  private String name;

  private String description;

  public PackageData(String name){

    this(name,"");

  }

  public PackageData(String name,String description) {

    this.name=name;

    this.description=description;

  }

  public String getName(){

    return name;

  }

  public String getDescription(){

    return description;

  }

}

                        list 2.6 PackageData.java

读者可能注意到这些类定义完全基于DTD文件,而且之前的XSLT也是基于DTD结构的,所以当处理的XML文档结构复杂时一定要定义DTD或XML Schema,即便不包含数据验证机制这样的需求,它们也有助于软件的开发。

  

References

[1] Eric M. Burke. Java and XSLT  O’Reilly&Associates,Inc. 2001

 

 

Copyright: Xingchen Chu@Copyright Reserved(c) 2004 Melbourne University

本文地址:http://com.8s8s.com/it/it16544.htm