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

原创 2004年05月11日 11:03:00

XMLXSLT实现代码生成器(IV

结果处理<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />

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

<?xml:namespace prefix = v ns = "urn:schemas-microsoft-com:vml" />CSDN_Dev_Image_2004-5-101958400.jpg

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: <?xml:namespace prefix = st1 ns = "urn:schemas-microsoft-com:office:smarttags" />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节点,通过JAXPTransformer对象实现从DOM节点并利用XSLTJava代码的转换。

CSDN_Dev_Image_2004-5-101958402.jpg

                                   Pic 2.3 UML for My Data Model and JDOM Util

数据模型

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

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

2.根据DTDELEMENT定义中的通配符,如果是*+则在对应元素的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文档结构复杂时一定要定义DTDXML Schema,即便不包含数据验证机制这样的需求,它们也有助于软件的开发。

 

 

References

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

 

 

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

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

XML和XSLT实现代码生成器(I)摘要       XML和XSLT为开发WEB应用提供了非常好的解决方案,然而,它们的能力不仅限制在WEB开发上,其实它们提供了很好的基于元数据(meta data...
  • starchu1981
  • starchu1981
  • 2004年04月28日 00:00
  • 2739

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

XML和XSLT实现代码生成器(II)XSLT处理元数据       如前文所述,当建立元数据以后,就可以使用XSLT将XML数据转换为实际的代码了,列表1.3展示了一个XSL文档,它将处理上述的XM...
  • starchu1981
  • starchu1981
  • 2004年04月28日 00:00
  • 2579

XML/XSD/XSLT

XSD(XML Schema Definition) XML Schema 是基于 XML 的 DTD 替代者。 XML Schema 描述 XML 文档的结构。 XML Schem...
  • lein_wang
  • lein_wang
  • 2012年07月30日 11:27
  • 1612

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

 XML和XSLT实现代码生成器(III) XSLT处理元数据(续)   —命名模板,打印set方法à                    public void set          —属性首...
  • starchu1981
  • starchu1981
  • 2004年04月27日 23:54
  • 2101

[XML]学习笔记(八)XSLT

一、XSL(eXtensible Stylesheet Language)扩展样式表语言:主要包含三个部分——XSLT用于XML文档转换,XPath用于在XML文档中导航,XSL-FO用于XML文档格...
  • CristianoJason
  • CristianoJason
  • 2016年05月20日 00:45
  • 4480

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

XML和XSLT实现代码生成器(V)完 生成JDOM文档       当我们定义好数据模型后,下一步就是将数据模型转换为JDOM文档结构。这里可以有不同的设计方式:可以为每个Java类定义对应的工具类...
  • starchu1981
  • starchu1981
  • 2004年05月07日 15:57
  • 1862

通过XSLT转换XML

一 介绍 XSLT是一种用来转换XML文档结构的语言,它是EXtensible Style Language Extensions Transformations的缩写。 XSLT类似与HTML中...
  • chengqiuming
  • chengqiuming
  • 2017年04月12日 10:47
  • 607

XSLT——XML样式表转换语言

XML样式表转换语言 1. XSLT是XSLTransformations的缩写,它是XSL的一个组成部分。 XSL(EXtensible StyleSheet)由三部分组成: –  XSLT。...
  • liaoqianwen123
  • liaoqianwen123
  • 2014年04月02日 22:42
  • 1973

js通过xmldom调用xslt对xml排序输出的一个简单例子.

 test.xmlxml version="1.0" encoding="utf-8" ?>xml-stylesheet type="text/xsl" href="Test.xsl"?>-->roo...
  • fcuandy
  • fcuandy
  • 2007年12月13日 23:59
  • 1272

xslt格式化输出xml文件的三种方法(转)

xslt格式化输出xml文件的三种方法XSL由两部分组成: 一是转化XML文档;二是格式化XML文档。如果你不理解这个意思,可以这样想:XSL是一种可以将XML转化成HTML的语言,一种可以过滤和选择...
  • a651944226
  • a651944226
  • 2010年10月25日 17:04
  • 1610
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:XML和XSLT实现代码生成器(IV)
举报原因:
原因补充:

(最多只允许输入30个字)