Chinaunix首页 | 论坛 | 博客
  • 博客访问: 639870
  • 博文数量: 125
  • 博客积分: 8703
  • 博客等级: 中将
  • 技术积分: 1102
  • 用 户 组: 普通用户
  • 注册时间: 2010-03-10 17:48
文章分类

全部博文(125)

文章存档

2012年(2)

2011年(3)

2010年(11)

2009年(1)

2008年(12)

2007年(58)

2006年(38)

分类: Java

2007-08-24 16:30:12

一、关于XML解析
  XML在Java应用程序里变得越来越重要, 广泛应用于数据存储和交换. 比如我们常见的配置文件,都是以XML方式存储的. XML还应用于Java Message Service和Web Services等技术作为数据交换.因此,正确读写XML文档是XML应用的基础.
  Java提供了SAX和DOM两种方式用于解析XML,但即便如此,要读写一个稍微复杂的XML,也不是一件容易的事.
二、XMLBean简介
  Hibernate已经成为目前流行的面向Java环境的对象/关系数据库映射工具.在Hibernate等对象/关系数据库映射工具出现之前,对数据库的操作是通过JDBC来实现的,对数据库的任何操作,开发人员都要自己写SQL语句来实现. 对象/关系数据库映射工具出现后,对数据库的操作转成对JavaBean的操作,极大方便了数据库开发. 所以如果有一个类似的工具能够实现将对XML的读写转成对JavaBean的操作,将会简化XML的读写,即使对XML不熟悉的开发人员也能方便地读写XML. 这个工具就是XMLBean.
三、准备XMLBean和XML文档
  XMLBean是Apache的一个开源项目,可以从下载,最新的版本是2.0. 解压后目录如下:
xmlbean2.0.0   
 +---bin   
 +---docs   
 +---lib  
 +---samples   
 +---schemas
bin 目录里是一些命令,我们就是使用这些命令来生成xmlbean
为了演示一个例子,准备一个XML文档(customers.xml),
  在本文的例子里,我们将对这个文档进行读写操作. 文档源码如下:



   
        1
        female
        Jessica
        Lim
        1234567
       

           
                350106
                #25-1
                SHINSAYAMA 2-CHOME
           

           
                Ms Danielle
                350107
                #167
                NORTH TOWER HARBOUR CITY
           

       

   

   
        2
        male
        David
        Bill
        808182
       

           
                319087
                1033 WS St.
                Tima Road
           

           
                Mr William
                672993
                1033 WS St.
                Tima Road
           

       

   



这是一个客户的数据模型,每个客户都有客户编号(ID),姓名,性别(gender),电话号码(phoneNumber)和地址,其中地址有两个: 首要地址(PrimaryAddress)和帐单地址(BillingAddress),每个地址有邮编,地址1,和地址2组成.其中帐单地址还有收件人(receiver).

四、XMLBean使用步骤
  和其他面向Java环境的对象/关系数据库映射工具的使用步骤一样,在正式使用XMLBean前,我们要作两个准备.
  1. 生成XML Schema文件
  什么是XML Schema文件?
    正常情况下,每个XML文件都有一个Schema文件,XML Schema文件是一个XML的约束文件,它定义了XML文件的结构和元素.以及对元素和结构的约束. 通俗地讲,如果说XML文件是数据库里的记录,那么Schema就是表结构定义.
  为什么需要这个文件?
   XMLBean需要通过这个文件知道一个XML文件的结构以及约束,比如数据类型等. 利用这个Schema文件,XMLBean将会产生一系列相关的Java Classes来实现对XML的操作. 而作为开发人员,则是利用XMLBean产生的Java Classes来完成对XML的操作而不需要SAX或DOM.
   怎样产生这个Schema文件呢?
    如果对于熟悉XML的开发人员,可以自己来写这个Schema文件,对于不熟悉XML的开发人员,可以通过一些工具来完成.比较有名的如XMLSPY和Stylus Studio都可以通过XML文件来生成Schema文件. 加入我们已经生成这个Schema文件(customer.xsd):


  
   
       
           
                                    type="customerType" />
           

       

   

   
       
           
           
           
           
           
           
       

   

   
       
           
           
       

   


   
       
           
           
           
       

   

   
       
           
           
           
           
       

   



    2. 利用scomp来生成Java Classes
  scomp是XMLBean提供的一个编译工具,它在bin的目录下. 通过这个工具,我们可以将以上的Schema文件生成Java Classes.scomp的语法如下:-
  scomp [options] [dirs]* [schemaFile.xsd]* [service.wsdl]* [config.xsdconfig]*
  主要参数说明:
  -src [dir] -- 生成的Java Classes存放目录
  -srconly -- 不编译Java Classes,不产生Jar文件
  -out [jarFileName] -- 生成的Jar文件,缺省是xmltypes.jar
  -compiler -- Java编译器的路径,即Javac的位置
  schemaFile.xsd -- XML Schema文件位置
    config.xsdconfig -- xsdconfig文件的位置, 这个文件主要用来制定生成的Java Class的一些文件名规则和Package的名称,在本文,package是sample.xmlbean
准备一个配置文件(文件名customer.xsdconfig),它的内容如下:


 
    sample.xmlbean
 



    
  在本文,我是这样运行的:
      scomp -src src  -out customerXmlBean.jar customer.xsd    -compiler "C:\Program Files\Java\jdk1.6.0\bin\javac" customer.xsdconfig
  这个命令行的意思是告诉scomp生成customerXmlBean.jar,放在build目录下,同时生成源代码放在build\src下, Schema文件是customer.xsd,xsdconfig文件是customer.xsdconfig.其实, 生成的Java源代码没有多大作用,我们要的是jar文件.我们先看一下src\sample\xmlbean下生成的Classes.
  CustomersDocument.java -- 整个XML文档的Java Class映射
  CustomerType.java -- 节点sustomer的映射
  AddressType.java -- 节点address的映射
  BillingAddressType.java -- 节点billingAddress的映射
  PrimaryAddressType.java -- 节点primaryAddress的映射
    好了,到此我们所有的准备工作已经完成了. 下面就开始进入重点:利用刚才生成的jar文件读写XML.
   
五、利用XMLBean读XML文件
  新建一个Java Project,将XMLBean2.0.0\lib\下的Jar文件和刚才我们生成的customerXmlBean.jar加入到Project的ClassPath.
  新建一个Java Class: CustomerXMLBean. 源码如下:
    package com.sample.reader;

   
import java.io.File;
import sample.xmlbean.*;
import org.apache.xmlbeans.XmlOptions;

public class CustomerXMLBean {

    private String filename = null;

    public CustomerXMLBean(String filename) {
        super();
        this.filename = filename;
    }

    public void customerReader() {
        try {
            File xmlFile = new File(filename);
            CustomersDocument doc = CustomersDocument.Factory.parse(xmlFile);
            CustomerType[] customers = doc.getCustomers().getCustomerArray();

            for (int i = 0; i < customers.length; i++) {
                CustomerType customer = customers[i];
                println("Customer#" + i);
                println("Customer ID:" + customer.getId());
                println("First name:" + customer.getFirstname());
                println("Last name:" + customer.getLastname());
                println("Gender:" + customer.getGender());
                println("PhoneNumber:" + customer.getPhoneNumber());
                // Primary address
                PrimaryAddressType primaryAddress = customer.getAddress()
                        .getPrimaryAddress();
                println("PrimaryAddress:");
                println("PostalCode:" + primaryAddress.getPostalCode());
                println("AddressLine1:" + primaryAddress.getAddressLine1());
                println("AddressLine2:" + primaryAddress.getAddressLine2());
                // Billing address
                BillingAddressType billingAddress = customer.getAddress()
                        .getBillingAddress();
                println("BillingAddress:");
                println("Receiver:" + billingAddress.getReceiver());
                println("PostalCode:" + billingAddress.getPostalCode());
                println("AddressLine1:" + billingAddress.getAddressLine1());
                println("AddressLine2:" + billingAddress.getAddressLine2());
            }
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }

    private void println(String str) {
        System.out.println(str);
    }

    public static void main(String[] args) {
        String filename = "F://JavaTest//Eclipse//XMLBean//xml//customers.xml";
        CustomerXMLBean customerXMLBean = new CustomerXMLBean(filename);
        customerXMLBean.customerReader();
    }
}
运行它,参看输出结果:
 Customer#0
Customer ID:1
First name:Jessica
Last name:Lim
Gender:female
PhoneNumber:1234567
PrimaryAddress:
PostalCode:350106
AddressLine1:#25-1
AddressLine2:SHINSAYAMA 2-CHOME
BillingAddress:
Receiver:Ms Danielle
PostalCode:350107
AddressLine1:#167
AddressLine2:NORTH TOWER HARBOUR CITY
Customer#1
Customer ID:2
First name:David
Last name:Bill
Gender:male
PhoneNumber:808182
PrimaryAddress:
PostalCode:319087
AddressLine1:1033 WS St.
AddressLine2:Tima Road
BillingAddress:
Receiver:Mr William
PostalCode:672993
AddressLine1:1033 WS St.
AddressLine2:Tima Road

  怎么样,是不是很轻松? XMLBean的威力.
六、利用XMLBean写XML文件
  利用XMLBean创建一个XML文档也是一件轻而易举的事.我们再增加一个Method,
  请看一下的Java Class:
   public void createCustomer() {
        try { // Create Document
            CustomersDocument doc = CustomersDocument.Factory.newInstance();
            // Add new customer
            CustomerType customer = doc.addNewCustomers().addNewCustomer();
            // set customer info
            customer.setId(3);
            customer.setFirstname("Jessica");
            customer.setLastname("Lim");
            customer.setGender("female");
            customer.setPhoneNumber("1234567");
            // Add new address
            AddressType address = customer.addNewAddress();
            // Add new PrimaryAddress
            PrimaryAddressType primaryAddress = address.addNewPrimaryAddress();
            primaryAddress.setPostalCode("350106");
            primaryAddress.setAddressLine1("#25-1");
            primaryAddress.setAddressLine2("SHINSAYAMA 2-CHOME");

            // Add new BillingAddress
            BillingAddressType billingAddress = address.addNewBillingAddress();
            billingAddress.setReceiver("Ms Danielle");
            billingAddress.setPostalCode("350107");
            billingAddress.setAddressLine1("#167");
            billingAddress.setAddressLine2("NORTH TOWER HARBOUR CITY");
            File xmlFile = new File(filename);
            doc.save(xmlFile);
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }
  修改main method.
   public static void main(String[] args) {
        String filename = "F://JavaTest//Eclipse//XMLBean//xml//customers_new.xml";
        CustomerXMLBean customerXMLBean = new CustomerXMLBean(filename);
        customerXMLBean.createCustomer();
    }
  运行,打开customers_new.xml:
  

   
        3
        female
        Jessica
        Lim
        1234567
       

           
                350106
                #25-1
                SHINSAYAMA 2-CHOME
           

           
                Ms Danielle
                350107
                #167
                NORTH TOWER HARBOUR CITY
           

       

   



七、利用XMLBean修改XML文件
  我们再增加一个Method:
      public void updateCustomer(int id, String lastname) {
        try {
            File xmlFile = new File(filename);
            CustomersDocument doc = CustomersDocument.Factory.parse(xmlFile);
            CustomerType[] customers = doc.getCustomers().getCustomerArray();
            for (int i = 0; i < customers.length; i++) {
                CustomerType customer = customers[i];
                if (customer.getId() == id) {
                    customer.setLastname(lastname);
                    break;
                }
            }
            doc.save(xmlFile);
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }
  main method:
    public static void main(String[] args) { 
      String filename = "F://JavaTest//Eclipse//XMLBean//xml//customers_new.xml";
      CustomerXMLBean customerXMLBean = new   CustomerXMLBean(filename);
      customerXMLBean.updateCustomer(3,"last");
   }
  运行之后,我们将会看到客户编号为3的客户的lastname已经改为last.
八、利用XMLBean删除一个customer
  再增加一个Method:
    public void deleteCustomer(int id) {     try {      File xmlFile = new File(filename);     CustomersDocument doc = CustomersDocument.Factory.parse(xmlFile);    CustomerType[] customers = doc.getCustomers().getCustomerArray();
   for (int i = 0; i < customers.length; i++) {        CustomerType customer = customers[i];        if(customer.getId()==id){                        customer.setNil() ;                        break;               }   }   doc.save(xmlFile);   } catch (Exception ex) {        ex.printStackTrace();        }   }
  main method:
    public static void main(String[] args) {    String filename = "F://JavaTest//Eclipse//XMLBean//xml//customers_new.xml";                        CustomerXMLBean customerXMLBean = new CustomerXMLBean(filename);                        customerXMLBean.deleteCustomer(3);    }
  运行,我们将会看到客户编号为3的客户的资料已经被删除.
九、*.xsdconfig 更多用法
这里只是简单介绍一个用法,其他用法,可以参考 xmlbeans 目录里 Sample/xsdconfig 目录里的例子。
在应用中,我们可能会解析许多种xml文件,但是我们的系统中又不想引入太多额外的jar文件,因此,一个xml文件生成一个jar文件就显得不好了,好的方式是所有的xml class 都放到一个jar里,并且组织在各自的包里。
针对这种需求,我们来做一个例子:
首先写 .xsd 文件.如果不熟悉,xsd 文件建议使用工具生成(比如xmlspy),当然前提得了解一下xsd的基本知识。比如说下面的 xsd 文件


    xmlns:xs=""
    xmlns:ct=""
    targetNamespace=""
    elementFormDefault="qualified">
   
       
           
               
               
               
               
           

       

   

   
       
           
           
           
       

   

   
   
   
   

    xmlns:ct=""
    targetNamespace=""    这个是我们给这个 xsd 文件起的命名空间的名字。
        注意这个(红色部分):    如果你定义了命名空间,那么如果要使用 ref ,就必须加上命名空间的名字,否则使用scomp的    命令的时候,会提示找不到 element.
下面我们修改一下 xsdconfig 文件
    xmlns:ct="" xmlns:test2="">

 
    sample.testxmlbean
 

   
   
    sample.xmlbean
 

   
   
   
    为了    举例子说明在 xsdconfig 文件里可以使用很多个命名空间,这里加了其他一个命名空间作为例子,也就是蓝色字部分。通过这样修改 xsdconfig 文件,我们就可以为不同的xsd文件,生成不同的包的结构。
    一个 uri 也可以包含很多个命名空间,比如:
   

使用 scomp 命令scomp –src build/src –out build/test.jar test.xsd test2.xsd test.xsdconfigscomp 命令可以加很多个 xsd 文件作为参数。
十、下载资源
XmlSpy :
XmlBeans:
          xmlbeans-current.zip

   
十一、附加内容
a> Schema 教程
最简单的Schema文档如何写一个最简单的XML Schema文档呢? 首先,我们写出一个最简单的XML文档。hello.xml------------------- Hello World!!hello.xsd---------- XML Schema文档后缀名是.xsd,完全符合XML语法,根元素是schema,命名空间xmlns:xsd=",用元素定义实例文档中的元素,如greeting。2. 含子元素的Schema文档 假设实例文档是如下的:customer.xml-----------teiki
No.237, Road Waitan, Shanghai
则可以写出以下的XML Schema文档:customer.xsd----------------实例文档customer.xml中,元素含有两个子元素,所以我们在Schema文档中采用ComplexType来定义该元素。sequence表示子元素依次出现的顺序。3. 含子元素和孙元素的Schema文档 这次我们给出一个更加复杂一些的文档:customer.xml---------------Teiki
ZhejiangHangzhouXilu Road, No.121, 7F
为此,我们需要一个更加复杂一点的Schema文档:address.xsd-----------------不过,我们还可以采用ref元素来重新编写这个Schema文档:address2.xsd----------------------使用ref元素可以直接将其指向另一个模块,使文档更加具有可读性。
定义相同子元素的数量先看这个简单的订购数据实例文档:order.xml---------Accounting BookTaxation Book假设元素,即每次的订购书目不能超过10种,那该怎么写这个Schema文档呢?这里要用到的maxOccurs属性。order.xsd--------------------第7行中的maxOccurs属性为10,代表orderItem元素可以最大有10个。如果,不设定元素个数,则可以用maxOccurs="unbounded"来定义。类似,如果要定义最小值,可以使用minOccurs,比如下面这句:这两个属性缺省值都是1。5. 定义可选项的子元素假如上面的订书数据中,可以用书名或者书号任一一种订购,则实例文档可能如下:order2.xml-----------------Accounting Book7-5058-3496-7这时书写Schema文档还需要使用choice元素。order2.xsd-------------------------稍微更复杂的可选项子元素 再稍微修改一下订书数据的实例文档:order3.xml-----------------Accounting Book27-5058-3496-7这里假定值为1时,缺省。如何修改Schema文档呢?order3.xsd-----------------19行中的quantity最少出现值为0,也就是可以有,也可以没有。当然,也可以直接在元素中,包含quantity,然后定义它的minOccurs。
内置简单类型图省略7. 自定义简单类型 如果内置简单类型的44种还不能满足要求,怎么办呢?下面学习自定义简单类型。(XML的扩展性充分体现在这里)例如这个实例文档:order4.xml-----------------7-5058-3496-75ID是一个标准的ISBN编码,我们怎么定义这个ISBN编码呢?idType是一个自定义的简单类型。我们对它做了限制:代表它是基于一个字符串类型。再用pattern元素来描述该字符串的形式。value="\d{1}-\d{4}-\d{4}-\d{1}"这是一个正则表达式,关于正则表达式,以后再介绍。嘻嘻!利用这个自定义的简单类型,我们可以重新写Schema文档:order4.xsd---------------假如我们事先确定好ID只有3个,即只有3个ISBN是可选的,那怎么办?我们可以用enumeration元素来进行列举。再来看订购量quantity的值,如果我们设定其值必须在1-10之间,该怎么办呢?可以这些自定义一个简单类型。其中,minInclusive,maxInclusive分别代表该类型的取值范围。所以最终修改后的Schema文档如下:order4-1.xsd----------------------

8. 定义属性 最后,我们再来讲讲元素的属性如何在Schema文档中定义。比如上面的order.xml实例文档中:对此,我们在Schema文档中采用一个attribute来定义:order.xsd---------  ←空元素  那么,实例文档中该属性值是必须的还是可有可无的呢?我们可以这样限制:这里我们讲id属性类型作为一种自定义数据类型idType。而且,用attribute元素的use属性来定义是否是必须的属性。required是必须值,optional是可选值,prohibited是无属性值。那么对于属性的缺省值,我们怎么定义呢?比如:我们还可以用attribute元素的另一个属性default来定义:所以,我们可以重新写出一个Schema文档:order2.xsd--------------上面的属性我们定义我们还可以采用属性组的办法来重新改写Schema文档。 order3.xsd----------------这个属性组就不详细解释了,不过,大家一看就清楚了吧。最后,我们写一个完整的订书order.xml的Schema文档。其他xml中引用xsdXsd中对应为:ElementFormDefault=”qualified” 意思是要求element使用的namespace是targetNamespace,它的作用是对元素起"限定与非限定"使用,意思是在文档范例中要求采用命名空间前缀。
 
阅读(2289) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~