XStream在JavaBean与XML/JSON 之间相互转换

时间:2022-01-06 19:05:14

原址:http://kingxss.iteye.com/blog/1037741

1.XStream:http://xstream.codehaus.org/ 
2.API:http://xstream.codehaus.org/javadoc/index.html

 

    XML 和 JSON 是当今常用的两种数据描述与传输的格式,特别是涉及到 JS 时使用 JSON 颇为频繁。自然,在 Java  的世界里少不了完成 JavaBean 与这两种格式相互转换的组件,那就是 XStream 和 JSON-lib。这里简单记下 XStream  的用法。

     其实相类似的工具早已有之。如果用过 DWR 的同志,一定有印像,DWR 进行远程方法调用时也能为你完成 JavaBean 和 JSON 格式的双向转换的,所依赖的是它的各种 Converter。再要是对 Struts1 的细节有所注意的话,Struts1 的 ActionServlet 在初始化 struts-config.xml 时是通过 commons-digester  来完成 XML 到 JavaBean 转换的。相应的 Apache 也有一个 commons-betwixt  实现了 JavaBean 到 XML 的生成。

     而这里要说的 XStream(http://xstream.codehaus.org ) 把 JavaBean 与 XML/JSON 间的双向转换统统实现了。要使用 XStream,需下载到xstream 包,当前版本是 1.4.3。1.4.*对1.3.*进行了大幅度的功能提升。然后把 xstream-1.x.x.jar 添加到项目的 Classpath 上,可不依赖于其他包。在某些有要求时候才需要用到 lib 目录中的其他包,下面会提到。

简单说明 XStream 的使用吧,分为 JavaBean -> XML、JavaBean -> JSON、 XML -> JavaBean、JSON -> JavaBean 几部分内容。在开始例子之前,先定义三个类(都在 com.unmi.model 包中):

Java代码  XStream在JavaBean与XML/JSON 之间相互转换
  1. public class Customer {  
  2.     private int custId;  
  3.     private String custName;  
  4.     private List<Order> orders;  
  5.     //setter/getter 和构造方法略  
  6. }  
  7.    
  8. public class Order {  
  9.     private int orderId;  
  10.     private String orderName;  
  11.     private Product[] products;  
  12.     //setter/getter 和构造方法略  
  13. }  
  14.    
  15. public class Product {  
  16.     private int prodId;  
  17.     private String prodName;  
  18.     private double prodPrice;  
  19.     //setter/getter 和构造方法略  
  20. }  

 Customer/Order/Product,它们之间的关系,依次是一对多、一对多的,为演示目的,分别用了 List 和数组作为聚合属性。

 

1. JavaBean -> XML

Java代码  XStream在JavaBean与XML/JSON 之间相互转换
  1. public static void main(String[] args) {  
  2.    
  3.     //构造接近实际的 Customer 对象  
  4.     Product p1 = new Product(1001,"电脑",4000);  
  5.     Product p2 = new Product(1002,"空调",2000);  
  6.     Product[] prods1 = new Product[]{p1,p2};  
  7.    
  8.     Order order1 = new Order(101,"电器类",prods1);  
  9.    
  10.     List<Order> orders = new ArrayList<Order>();  
  11.     orders.add(order1);  
  12.     Customer customer = new Customer(1,"Unmi",orders);  
  13.    
  14.     //XStream xstream = new XStream();  
  15.     XStream xstream = new XStream(new DomDriver());  
  16.    
  17.     String xml = xstream.toXML(customer);//转换成 xml 格式  
  18.    
  19.     System.out.println(xml); //输出 xml 字符串  
  20. }  

      代码说明:XStream 对象的构造,可无参,或传入某一 DomDriver 实例,如 XppDomDriver、JDomDriver、Dom4JDriver,它们的用途我想不必多说,注意要引入相应的 jar 包,无参或 DomDriver 则是用 JDK 默认的解析 XML 的实现。

toXML() 还有两个重载方法,分别是:toXML(Object obj, OutputStream out) 和  toXML(Object obj, Writer out),可用于自定义输出目的地。

来看看上面程序的输出:

Xml代码  XStream在JavaBean与XML/JSON 之间相互转换
  1. <com.unmi.model.Customer>  
  2.   <custId>1</custId>  
  3.   <custName>Unmi</custName>  
  4.   <orders>  
  5.     <com.unmi.model.Order>  
  6.       <orderId>101</orderId>  
  7.       <orderName>电器类</orderName>  
  8.       <products>  
  9.         <com.unmi.model.Product>  
  10.           <prodId>1001</prodId>  
  11.           <prodName>电脑</prodName>  
  12.           <prodPrice>4000.0</prodPrice>  
  13.         </com.unmi.model.Product>  
  14.         <com.unmi.model.Product>  
  15.           <prodId>1002</prodId>  
  16.           <prodName>空调</prodName>  
  17.           <prodPrice>2000.0</prodPrice>  
  18.         </com.unmi.model.Product>  
  19.       </products>  
  20.     </com.unmi.model.Order>  
  21.   </orders>  
  22. </com.unmi.model.Customer>  

应该发现了,节点名用了类的全限名,有些难看,不过我们可以用别名来解决,只要在 toXML() 之前加上三行代码:

Java代码  XStream在JavaBean与XML/JSON 之间相互转换
  1. xstream.alias("customer", Customer.class);  
  2. xstream.alias("order", Order.class);  
  3. xstream.alias("product", Product.class);  

执行,再来看看生成的 XML 内容,漂亮多了吧:

Xml代码  XStream在JavaBean与XML/JSON 之间相互转换
  1. <customer>  
  2.   <custId>1</custId>  
  3.   <custName>Unmi</custName>  
  4.   <orders>  
  5.     <order>  
  6.       <orderId>101</orderId>  
  7.       <orderName>电器类</orderName>  
  8.       <products>  
  9.         <product>  
  10.           <prodId>1001</prodId>  
  11.           <prodName>电脑</prodName>  
  12.           <prodPrice>4000.0</prodPrice>  
  13.         </product>  
  14.         <product>  
  15.           <prodId>1002</prodId>  
  16.           <prodName>空调</prodName>  
  17.           <prodPrice>2000.0</prodPrice>  
  18.         </product>  
  19.       </products>  
  20.     </order>  
  21.   </orders>  
  22. </customer>  

 

 

如果你想去掉<orders>和<products>这些简单的包装标签,只需要在只要在 toXML() 之前加上:

Java代码  XStream在JavaBean与XML/JSON 之间相互转换
  1. xstream.addImplicitCollection(Customer.class"orders");  
  2. xstream.addImplicitArray(Order.class"products");  

甚至可以将coustid和custName设置为coustomer的属性节点,只需要在只要在 toXML() 之前加上:

Java代码  XStream在JavaBean与XML/JSON 之间相互转换
  1. xstream.aliasAttribute(Customer.class"custId""id");  
  2. xstream.aliasAttribute(Customer.class"custName""name");  

通过上面的一些列设置后,其结果如下

Xml代码  XStream在JavaBean与XML/JSON 之间相互转换
  1. <customer id="1" name="Unmi">  
  2.     <order>  
  3.         <orderId>101</orderId>  
  4.         <orderName>电器类</orderName>  
  5.         <product>  
  6.             <prodId>1001</prodId>  
  7.             <prodName>电脑</prodName>  
  8.             <prodPrice>4000.0</prodPrice>  
  9.         </product>  
  10.         <product>  
  11.             <prodId>1002</prodId>  
  12.             <prodName>空调</prodName>  
  13.             <prodPrice>2000.0</prodPrice>  
  14.         </product>  
  15.     </order>  
  16. </customer>  
 

 

2. JavaBean - > JSON

前面 main() 方法中构造好 Customer 对象后的代码换成如下:

Java代码  XStream在JavaBean与XML/JSON 之间相互转换
  1. XStream xstream = new XStream(new JsonHierarchicalStreamDriver());  
  2. xstream.alias("customer", Customer.class);  
  3. xstream.alias("order", Order.class);  
  4. xstream.alias("product", Product.class);  
  5. xstream.toXML(customer, new PrintWriter(System.out));  

      代码说明:这里对于 XStream 实例只是构造时换成了 JsonHierarchicalStreamDriver 实例,也可以用 JettisonMappedXmlDriver(需要引入 jettison-1.x.x.jar 包)。别名机制与前面的情况是一样的。仍然用 toXML() 方法,没有 toJSON() 方法,统一了接口方法以,用起来却让人有些费解。   如果使用的是 JettisonMappedXmlDriver,你会看到输出的内容全在一行。

看输出:

Json代码  XStream在JavaBean与XML/JSON 之间相互转换
  1. {"customer": {  
  2.   "custId"1,  
  3.   "custName""Unmi",  
  4.   "orders": [  
  5.     {  
  6.       "orderId"101,  
  7.       "orderName""电器类",  
  8.       "products": [  
  9.         {  
  10.           "prodId"1001,  
  11.           "prodName""电脑",  
  12.           "prodPrice"4000.0  
  13.         },  
  14.         {  
  15.           "prodId"1002,  
  16.           "prodName""空调",  
  17.           "prodPrice"2000.0  
  18.         }  
  19.       ]  
  20.     }  
  21.   ]  
  22. }}   

     前面用于演示 JavaBean 到 XML/JSON 的转换的例子,还稍显复杂,涉及到了 List 和数组类型。其实 XStream 是通过反射来获取属性的,即使是私有的,而不依赖于 getter 方法,这点上比JSON-lib 要强。XStream 使用了像 JDBC Driver 的模式,通过更换 Driver 的方式来达成不同的内部实现。和 DWR/Struts 一样,它也是用 Converter 来做到数据类型的转换。

3. XML -> JavaBean

Java代码  XStream在JavaBean与XML/JSON 之间相互转换
  1. public static void main(String[] args) {  
  2.    
  3.     XStream xstream = new XStream(new DomDriver());  
  4.    
  5.     //设置了这个别名才能识别下面 xml 中的 product 节点,否则要用类全限名称  
  6.     xstream.alias("product", Product.class);  
  7.     String xml = "<product><prodId>1001</prodId><prodName>电脑" +  
  8.             "</prodName><prodPrice>4000.0</prodPrice></product>";  
  9.     Product prod = (Product)xstream.fromXML(xml);  
  10.     System.out.println(prod.getProdName());  
  11. }  

执行上面的程序,能够输出产品名称“电脑”来,说明,由 XML 描述创建对象是成功的。相对于生成 XML 用的是 toXML(),通过 XML 构建对象用的方法是 fromXML(),同样要留意它的其他重载方法:

Object fromXML(InputStream input);
Object fromXML(InputStream xml, Object root);
Object fromXML(Reader xml);
Object fromXML(Reader xml, Object root);
Object fromXML(String xml);
Object fromXML(String xml, Object root);

它们能从不同的输入源,可指定根节点来构建对象。

4. JSON -> JavaBean

Java代码  XStream在JavaBean与XML/JSON 之间相互转换
  1. public static void main(String[] args) {  
  2.     // 这里不能用 JsonHierarchicalStreamDriver 了,它只能用于 JavaBean->JSON  
  3.     XStream xstream = new XStream(new JettisonMappedXmlDriver());  
  4.    
  5.     // 设置了这个别名才能识别下面 xml 中的 product 节点,否则要用类全限名称  
  6.     xstream.alias("product", Product.class);  
  7.     String json = "{product:{prodId: 1001,prodName: '电脑', prodPrice: 4000.0}}";  
  8.     Product prod = (Product) xstream.fromXML(json);  
  9.     System.out.println(prod.getProdName());  
  10. }  

可以看到也输出了“电脑”,说明工作正常。注意这里只能用 JettisonMappedXmlDriver。

相比于 JavaBean 到 XML/JSON 的转换,下面两个例子要简单多了,JavaBean 未涉及到关联关系。复杂的 XStream 也做得到,就看前两个例子,XStream 能够把复杂的 JavaBean 生成 XML/JSON,它也有这个能耐把生成的东西还原回去的。而且 XStream 在由 XML/JSON 生成 JavaBean 时不依赖于 setter 方法和构造方法的。

我们在实际中使用 XStream 时应该还会对它进行细究,或作进一步的扩展,如把某个 JavaBean 属性生成 XML 时作为另一属性生成的 XML 节点的属性,而不是子节点;或加入自己的 Converter、甚至是自己的 DomDriver、JsonDriver 等等。

附件:使用上述方法完成XStream进行格式转换的工具类集合,包含了相应jar包。