WebService CXF学习(进阶篇4):JAXB剖析
?前面几节我们讲解对象传递,但是通常情况下我们不直接传对象,因为直接传递对象安全性差,而且暴露了实体对象。所以我们选择传递XML文件,当然也可以传递JSON对象。这节我只针对传递XML,那么JAVA绑定成XML,服务端将XML解析成Java对象有什么工具可用吗,其实这样的工具多的是。这里我选择一个比较简单的JAXB工具来讲解一下。?
??? JAXB(Java Architecture for XML Binding)提供了一个快速而方便的方式绑定XML Schemas和java,使java程序员能够很方便的在java应用程序中处理XML数据。JAXB提供了将XML文档解组为java内容树的方法,以及将java内容树重新编组回XML文档的方法。JAXB同样也提供了一种从java对象生成XML Schema的方式。?
??? 这里有几个重要的定义:?
????编组(Marshalling)是把内存中的数据转化到存储媒介上的过程。因此在 Java 和 XML 环境中,编组就是把一些 Java 对象转化成一个(或多个) XML 文档。在数据库环境中,则是把 Java 表示的数据存入数据库。显然,编组的秘密在于把 Java 实例中的面向对象结构转化成适用于 XML 的 扁平结构,或者 RDBMS 中的关系结构(使用 Java 技术转换到 OODBMS 实际上很简单)。工作原理如下图所示:?
?
????解组(Unmarshalling) 是把数据从存储媒介转换到内存中的过程--正好与编组相反。因此需要把 XML 文档解组到 Java VM 中。这里的复杂性不是在扁平数据中,因为这不是必需的,而在于从正确的数据到正确的 Java 代码变量的映射。如果映射是错误的,就不可能正确地访问数据。当然,如果再尝试重新编组还会造成更大的问题,并且问题传播得很快。工作原理如下图所示:?
?
????往返(Round-tripping)可能是最重要也最容易误解的数据绑定术语。往返用于描述从存储媒介到内存然后回到存储媒介的完整循 环。在 XML 和 Java 技术环境中,这就意味着从 XML 文档到 Java 实例变量,然后再回到 XML 文档。正确的往返要求,如果中间没有修改数据,XML 输入和 XML 输出应该是等同的。?
??? 下载地址:http://java.sun.com/developer/technicalArticles/WebServices/jaxb/?
???
??? 我们还以例子来说明它的工作原理,直观点。?
??? 第一步,创建一个Customer对象?
???

- @XmlRootElement(name="customer")??
- @XmlAccessorType(XmlAccessType.FIELD)??
- @XmlType(name?=?"")??
- public?class?Customer?{??
- ??
- ????@XmlAttribute(required?=?true)??
- ????protected?String?name;??
- ????@XmlAttribute(required?=?true)??
- ????protected?int?age;??
- ??
- ????/**?
- ?????*?Gets?the?value?of?the?name?property.?
- ?????*??
- ?????*?@return?
- ?????*?????possible?object?is?
- ?????*?????{@link?String?}?
- ?????*??????
- ?????*/??
- ????public?String?getName()?{??
- ????????return?name;??
- ????}??
- ??
- ????/**?
- ?????*?Sets?the?value?of?the?name?property.?
- ?????*??
- ?????*?@param?value?
- ?????*?????allowed?object?is?
- ?????*?????{@link?String?}?
- ?????*??????
- ?????*/??
- ????public?void?setName(String?value)?{??
- ????????this.name?=?value;??
- ????}??
- ??
- ????/**?
- ?????*?Gets?the?value?of?the?age?property.?
- ?????*??
- ?????*/??
- ????public?int?getAge()?{??
- ????????return?age;??
- ????}??
- ??
- ????/**?
- ?????*?Sets?the?value?of?the?age?property.?
- ?????*??
- ?????*/??
- ????public?void?setAge(int?value)?{??
- ????????this.age?=?value;??
- ????}??
- ??
- }??
- ??
- ??????
??? 第二步,创建一个测试类

- public?class?SoapClient?{??
- ??
- ????private?final?static?String?MODEL?=?"com.itdcl.model";??
- ????public?static?void?main(String[]?args)?throws?ParserConfigurationException,?JAXBException,?TransformerException{??
- ??
- ??????????
- ????????ObjectFactory?factory?=?new?ObjectFactory();??
- ????????Customer?customer?=?factory.createCustomer();??
- ????????customer.setAge(20);??
- ????????customer.setName("Josen");??
- ??
- ????????DocumentBuilderFactory?dbf?=?DocumentBuilderFactory.newInstance();??
- ????????dbf.setNamespaceAware(true);??
- ????????DocumentBuilder?db?=?dbf.newDocumentBuilder();??
- ????????Document?doc?=?db.newDocument();??
- ??
- ????????JAXBContext?jaxbContext?=?JAXBContext.newInstance(MODEL);??
- ????????//Java对象转换成XML??
- ????????????????Marshaller?marshaller?=?jaxbContext.createMarshaller();??
- ????????marshaller.marshal(customer,?doc);??
- ??????????
- ????????DOMSource?domSource?=?new?DOMSource(doc);??
- ????????StringWriter?writer?=?new?StringWriter();??
- ????????StreamResult?result?=?new?StreamResult(writer);??
- ????????TransformerFactory?tf?=?TransformerFactory.newInstance();??
- ????????Transformer?transformer?=?tf.newTransformer();??
- ????????transformer.transform(domSource,?result);??
- ????????String?xmlString?=?writer.toString();??
- ????????System.out.println(xmlString);??
- ????????//XML转换成Java对象??
- ????????Unmarshaller?unmarshaller?=?jaxbContext.createUnmarshaller();??
- ????????StringReader?reader?=?new?StringReader(xmlString);??
- ????????Customer?cus?=?(Customer)unmarshaller.unmarshal(reader);??
- ????????System.out.println("Age:"+cus.getAge());??
- ????????System.out.println("Name:"+cus.getName());??
- ??????????
- ??????????
- ????}??
- }??
????? 第三步,运行一个测试类,看看效果如何。Java与XML之间转换如此简单?
????? 编组操作:利用上面生成的java文件执行编组操作。

- JAXBContext?jaxbContext?=?JAXBContext.newInstance(MODEL);??
- //Java对象转换成XML??
- ??????????????Marshaller?marshaller?=?jaxbContext.createMarshaller();??
- marshaller.marshal(customer,?doc);??
- ??
- DOMSource?domSource?=?new?DOMSource(doc);??
- StringWriter?writer?=?new?StringWriter();??
- StreamResult?result?=?new?StreamResult(writer);??
- TransformerFactory?tf?=?TransformerFactory.newInstance();??
- Transformer?transformer?=?tf.newTransformer();??
- transformer.transform(domSource,?result);??
- String?xmlString?=?writer.toString();??
- System.out.println(xmlString);??
???? 解组操作:通过xml文件执行解组操作。

- ??????????????JAXBContext?jaxbContext?=?JAXBContext.newInstance(MODEL);??
- ??????????????Unmarshaller?unmarshaller?=?jaxbContext.createUnmarshaller();??
- StringReader?reader?=?new?StringReader(xmlString);??
- Customer?cus?=?(Customer)unmarshaller.unmarshal(reader);??
- System.out.println("Age:"+cus.getAge());??
- System.out.println("Name:"+cus.getName());??
??? 也可通过Ant配置来解组,如下:

- <?xml?version="1.0"?encoding="utf-8"??>??
- <project?default="xjc-compile"?basedir=".">??
- ????<property?name="src.dir"?location="src"?/>??
- ????<property?name="lib.dir"?location="E:/cxf-lib"?/>??
- ????<property?name="xml-schema.dir"?location="src/WEB-INF"?/>??
- ????<property?name="schema.name"?value="cxfdemo.xsd"?/>??
- ????<property?name="package"?value="com.itdcl.model"?/>??
- ??
- ????<path?id="classpath">??
- ????????<fileset?dir="${lib.dir}"?includes="*.jar"?/>??
- ????</path>??
- ????<taskdef?name="xjc"?classname="com.sun.tools.xjc.XJCTask"??
- ????????classpathref="classpath"?/>??
- ??????????
- ????<target?name="xjc-compile">??
- ????????<echo?message="Build?Jaxb?Class?from?Schema"?/>??
- ????????<xjc?schema="${xml-schema.dir}/${schema.name}"??
- ????????????destdir="${src.dir}"?package="${package}"?>??
- ????????????<produces?dir="src/com/itdcl/model"?includes="*"?/>??
- ????????</xjc>??
- ????</target>??
- </project> ?
?