`
frank1998819
  • 浏览: 764694 次
  • 性别: Icon_minigender_1
  • 来自: 南京
文章分类
社区版块
存档分类

XStream把xml文件转化为java对象(转)

    博客分类:
  • Java
 
阅读更多

原作者:http://www.blogjava.net/DLevin/archive/2012/11/30/392240.html

 

现在参与的项目是一个纯Application Server,整个Server都是自己搭建的,使用JMS消息实现客户端和服务器的交互,交互的数据格式采用XML。说来惭愧,开始为了赶进度,所有XML消息都是使用字符串拼接的,而XML的解析则是使用DOM方式查找的。我很早就看这些代码不爽了,可惜一直没有时间去重构,最近项目加了几个人,而且美国那边也开始渐渐的把这个项目开发的控制权交给我们了,所以我开始有一些按自己的方式开发的机会了。因而最近动手开始重构这些字符串拼接的代码。

XMLJava Bean的解析框架,熟悉一点的只有DigesterXStreamDigester貌似只能从XML文件解析成Java Bean对象,所以只能选择XStream来做了,而且同组的其他项目也有在用XStream。一直听说XStream的使用比较简单,而且我对ThoughtWorks这家公司一直比较有好感,所以还以为引入XStream不会花太多时间,然而使用以后才发现XStream并没有想象的你那么简单。不过这个也有可能是因为我不想改变原来的XML数据格式,而之前的XML数据格式的设计自然不会考虑到如何便利的使用XStream。因而记录在使用过程中遇到的问题,供后来人参考,也为自己以后如果打算开其源码提供参考。废话就到这里了,接下来步入正题。

首先对于简单的引用,XStream使用起来确实比较简单,比如自定义标签的属性、使用属性和使用子标签的定义等:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->@XStreamAlias("request")
publicclass XmlRequest1 {
privatestatic XStream xstream;
static {
xstream
=new XStream();
xstream.autodetectAnnotations(
true);
}

@XStreamAsAttribute
private String from;

@XStreamAsAttribute
@XStreamAlias(
"calculate-method")
private String calculateMethod;

@XStreamAlias(
"request-time")
private Date requestTime;

@XStreamAlias(
"input-files")
private List<InputFileInfo> inputFiles;

publicstatic String toXml(XmlRequest1 request) {
StringWriter writer
=new StringWriter();
writer.append(Constants.XML_HEADER);
xstream.toXML(request, writer);
return writer.toString();
}
publicstatic XmlRequest1 toInstance(String xmlContent) {
return (XmlRequest1)xstream.fromXML(xmlContent);
}

@XStreamAlias(
"input-file")
publicstaticclass InputFileInfo {
private String type;
private String fileName;

}
publicstaticvoid main(String[] args) {
XmlRequest1 request
= buildXmlRequest();
System.out.println(XmlRequest1.toXml(request));
}
privatestatic XmlRequest1 buildXmlRequest() {

}
}

对以上Request定义,我们可以得到如下结果:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--><?xml version="1.0" encoding="UTF-8"?>
<request from="levin@host" calculate-method="advanced">
<request-time>2012-11-28 17:11:54.664 UTC</request-time>
<input-files>
<input-file>
<type>DATA</type>
<fileName>data.2012.11.29.dat</fileName>
</input-file>
<input-file>
<type>CALENDAR</type>
<fileName>calendar.2012.11.29.dat</fileName>
</input-file>
</input-files>
</request>

可惜这个世界不会那么清净,这个格式有些时候貌似并不符合要求,比如request-time的格式、input-files的格式,我们实际需要的格式是这样的:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--><?xml version="1.0" encoding="UTF-8"?>
<request from="levin@host" calculate-method="advanced">
<request-time>20121128T17:51:05</request-time>
<input-file type="DATA">data.2012.11.29.dat</input-file>
<input-file type="CALENDAR">calendar.2012.11.29.dat</input-file>
</request>

对不同Date格式的支持可以是用Converter实现,在XStream中默认使用自己实现的DateConverter,它支持的格式是:yyyy-MM-dd HH:mm:ss.S 'UTC',然而我们现在需要的格式是yyyy-MM-dd’T’HH:mm:ss,如果使用XStream直接注册DateConverter,可以使用配置自己的DateConverter,但是由于DateConverter的构造函数的定义以及@XStreamConverter的构造函数参数的支持方式的限制,貌似DateConverter不能很好的支持注解方式的注册,因而我时间了一个自己的DateConverter以支持注解:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->publicclass LevinDateConverter extends DateConverter {
public LevinDateConverter(String dateFormat) {
super(dateFormat, new String[] { dateFormat });
}
}

requestTime字段中需要加入以下注解定义:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->@XStreamConverter(value=LevinDateConverter.class, strings={"yyyyMMdd'T'HH:mm:ss"})
@XStreamAlias(
"request-time")
private Date requestTime;

对集合类,XStream提供了@XStreamImplicit注解,以将集合中的内容摊平到上一层XML元素中,其中itemFieldName的值为其使用的标签名,此时InputFileInfo类中不需要@XStreamAlias标签的定义:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->@XStreamImplicit(itemFieldName="input-file")
private List<InputFileInfo> inputFiles;

InputFileInfo中的字段,type作为属性很容易,只要为它加上@XStreamAsAttribute注解即可,而将fileName作为input-file标签的一个内容字符串,则需要使用ToAttributedValueConverter,其中Converter的参数为需要作为字符串内容的字段名:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->@XStreamConverter(value=ToAttributedValueConverter.class, strings={"fileName"})
publicstaticclass InputFileInfo {
@XStreamAsAttribute
private String type;
private String fileName;

}

XStream对枚举类型的支持貌似不怎么好,默认注册的EnumSingleValueConverter只是使用了Enum提供的name()和静态的valueOf()方法将enum转换成String或将String转换回enum。然而有些时候XML的字符串和类定义的enum值并不完全匹配,最常见的就是大小写的不匹配,此时需要写自己的Converter。在这种情况下,我一般会在enum中定义一个name属性,这样就可以自定义enum的字符串表示。比如有TimePeriodenum

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->publicenum TimePeriod {
MONTHLY(
"monthly"), WEEKLY("weekly"), DAILY("daily");

private String name;

public String getName() {
return name;
}

private TimePeriod(String name) {
this.name = name;
}

publicstatic TimePeriod toEnum(String timePeriod) {
try {
return Enum.valueOf(TimePeriod.class, timePeriod);
}
catch(Exception ex) {
for(TimePeriod period : TimePeriod.values()) {
if(period.getName().equalsIgnoreCase(timePeriod)) {
return period;
}
}
thrownew IllegalArgumentException("Cannot convert <"+ timePeriod +"> to TimePeriod enum");
}
}
}

我们可以编写以下Converter以实现对枚举类型的更宽的容错性:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->publicclass LevinEnumSingleNameConverter extends EnumSingleValueConverter {
privatestaticfinal String CUSTOM_ENUM_NAME_METHOD ="getName";
privatestaticfinal String CUSTOM_ENUM_VALUE_OF_METHOD ="toEnum";

private Class<?extends Enum<?>> enumType;

public LevinEnumSingleNameConverter(Class<?extends Enum<?>> type) {
super(type);
this.enumType = type;
}

@Override
public String toString(Object obj) {
Method method
= getCustomEnumNameMethod();
if(method ==null) {
returnsuper.toString(obj);
}
else {
try {
return (String)method.invoke(obj, (Object[])null);
}
catch(Exception ex) {
returnsuper.toString(obj);
}
}
}

@Override
public Object fromString(String str) {
Method method
= getCustomEnumStaticValueOfMethod();
if(method ==null) {
return enhancedFromString(str);
}
try {
return method.invoke(null, str);
}
catch(Exception ex) {
return enhancedFromString(str);
}
}

private Method getCustomEnumNameMethod() {
try {
return enumType.getMethod(CUSTOM_ENUM_NAME_METHOD, (Class<?>[])null);
}
catch(Exception ex) {
returnnull;
}
}

private Method getCustomEnumStaticValueOfMethod() {
try {
Method method
= enumType.getMethod(CUSTOM_ENUM_VALUE_OF_METHOD, (Class<?>[])null);
if(method.getModifiers() == Modifier.STATIC) {
return method;
}
returnnull;
}
catch(Exception ex) {
returnnull;
}
}

private Object enhancedFromString(String str) {
try {
returnsuper.fromString(str);
}
catch(Exception ex) {
for(Enum<?> item : enumType.getEnumConstants()) {
if(item.name().equalsIgnoreCase(str)) {
return item;
}
}
thrownew IllegalStateException("Cannot converter <"+ str +"> to enum <"+ enumType +">");
}
}
}

如下方式使用即可:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->@XStreamAsAttribute
@XStreamAlias(
"time-period")
@XStreamConverter(value
=LevinEnumSingleNameConverter.class)
private TimePeriod timePeriod;

double类型,貌似默认的DoubleConverter实现依然不给力,它不支持自定义的格式,比如我们想在序列化的时候用一下格式: ###,##0.0########,此时又需要编写自己的Converter

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->publicclass FormatableDoubleConverter extends DoubleConverter {
private String pattern;
private DecimalFormat formatter;

public FormatableDoubleConverter(String pattern) {
this.pattern = pattern;
this.formatter =new DecimalFormat(pattern);
}

@Override
public String toString(Object obj) {
if(formatter ==null) {
returnsuper.toString(obj);
}
else {
return formatter.format(obj);
}
}

@Override
public Object fromString(String str) {
try {
returnsuper.fromString(str);
}
catch(Exception ex) {
if(formatter !=null) {
try {
return formatter.parse(str);
}
catch(Exception e) {
thrownew IllegalArgumentException("Cannot parse <"+ str +"> to double value", e);
}
}
thrownew IllegalArgumentException("Cannot parse <"+ str +"> to double value", ex);
}
}

public String getPattern() {
return pattern;
}
}

使用方式和之前的Converter类似:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->@XStreamAsAttribute
@XStreamConverter(value
=FormatableDoubleConverter.class, strings={"###,##0.0########"})
privatedouble value;

最后,还有两个XStream没法实现的,或者说我没有找到一个更好的实现方式的场景。第一种场景是XStream不能很好的处理对象组合问题:

在面向对象编程中,一般尽量的倾向于抽取相同的数据成一个类,而通过组合的方式构建整个数据结构。比如Student类中有nameaddressAddress是一个类,它包含citycodestreet等信息,此时如果要对Student对象做如下格式序列化:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--><student name=”Levin”>
<city>shanghai</city>
<street>zhangjiang</street>
<code>201203</code>
</student>

貌似我没有找到可以实现的方式,XStream能做是在中间加一层address标签。对这种场景的解决方案,一种是将Address中的属性平摊到Student类中,另一种是让Student继承自Address类。不过貌似这两种都不是比较理想的办法。

第二种场景是XStream不能很好的处理多态问题:

比如我们有一个Trade类,它可能表示不同的产品:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->publicclass Trade {
private String tradeId;
private Product product;

}
abstractclass Product {
private String name;
public Product(String name) {
this.name = name;
}

}
class FX extends Product {
privatedouble ratio;
public FX() {
super("fx");
}

}
class Future extends Product {
privatedouble maturity;
public Future() {
super("future");
}

}

通过一些简单的设置,我们能得到如下XML格式:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--><trades>
<trade trade-id="001">
<product class="levin.xstream.blog.FX" name="fx" ratio="0.59"/>
</trade>
<trade trade-id="002">
<product class="levin.xstream.blog.Future" name="future" maturity="2.123"/>
</trade>
</trades>

作为数据文件,对Java类的定义显然是不合理的,因而简单一些,我们可以编写自己的Converterclass属性从product中去除:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->xstream.registerConverter(new ProductConverter(
xstream.getMapper(), xstream.getReflectionProvider()));

public ProductConverter(Mapper mapper, ReflectionProvider reflectionProvider) {
super(mapper, reflectionProvider);
}

@Override
publicboolean canConvert(@SuppressWarnings("rawtypes") Class type) {
return Product.class.isAssignableFrom(type);
}

@Override
protected Object instantiateNewInstance(HierarchicalStreamReader reader, UnmarshallingContext context) {
Object currentObject
= context.currentObject();
if(currentObject !=null) {
return currentObject;
}

String name
= reader.getAttribute("name");
if("fx".equals(name)) {
return reflectionProvider.newInstance(FX.class);
}
elseif("future".equals(name)) {
return reflectionProvider.newInstance(Future.class);
}
thrownew IllegalStateException("Cannot convert <"+ name +"> product");
}
}

在所有Production上定义@XStreamAlias(“product”)注解。这时的XML输出结果为:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--><trades>
<trade trade-id="001">
<product name="fx" ratio="0.59"/>
</trade>
<trade trade-id="002">
<product name="future" maturity="2.123"/>
</trade>
</trades>

然而如果有人希望XML的输出结果如下呢?

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--><trades>
<trade trade-id="001">
<fx ratio="0.59"/>
</trade>
<trade trade-id="002">
<future maturity="2.123"/>
</trade>
</trades>

大概找了一下,可能可以定义自己的Mapper来解决,不过XStream的源码貌似比较复杂,没有时间深究这个问题,留着以后慢慢解决吧。

补充:

Map类型数据,XStream默认使用以下格式显示:

<map class="linked-hash-map">
<entry>
<string>key1</string>
<string>value1</string>
</entry>
<entry>
<string>key2</string>
<string>value2</string>
</entry>
</map>

 

但是对一些简单的Map,我们希望如下显示:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--><map>
<entry key="key1" value="value1"/>
<entry key="key2" value="value2"/>
</map>

 

对这种需求需要通过编写Converter解决,继承自MapConverter,覆盖以下函数,这里的Map默认keyvalue都是String类型,如果他们不是String类型,需要另外添加逻辑:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->@SuppressWarnings("rawtypes")
@Override
publicvoid marshal(Object source, HierarchicalStreamWriter writer,
MarshallingContext context) {
Map map
= (Map) source;
for (Iterator iterator = map.entrySet().iterator(); iterator.hasNext();) {
Entry entry
= (Entry) iterator.next();
ExtendedHierarchicalStreamWriterHelper.startNode(writer, mapper()
.serializedClass(Map.Entry.
class), entry.getClass());

writer.addAttribute(
"key", entry.getKey().toString());
writer.addAttribute(
"value", entry.getValue().toString());
writer.endNode();
}
}

@Override
@SuppressWarnings({
"unchecked", "rawtypes" })
protectedvoid putCurrentEntryIntoMap(HierarchicalStreamReader reader,
UnmarshallingContext context, Map map, Map target) {
Object key
= reader.getAttribute("key");
Object value
= reader.getAttribute("value");

target.put(key, value);
}

 

但是只是使用Converter,得到的结果多了一个class属性:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--><map class="linked-hash-map">
<entry key="key1" value="value1"/>
<entry key="key2" value="value2"/>
</map>

 

XStream中,如果定义的字段是一个父类或接口,在序列化是会默认加入class属性以确定反序列化时用的类,为了去掉这个class属性,可以定义默认的实现类来解决(虽然感觉这种解决方案不太好,但是目前还没有找到更好的解决方案)。

 

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->xstream.addDefaultImplementation(LinkedHashMap.class, Map.class);
分享到:
评论

相关推荐

    xstream将xml文档转换成json对象

    在这个场景中,我们将探讨如何使用XStream将XML文档转换成Java对象,进而转化为JSON对象。 首先,我们需要引入XStream库。XStream的核心功能是能够将Java对象和XML之间的映射自动化,极大地简化了序列化和反序列化...

    xstream轻松解析xml到java对象

    以下是一个简单的示例,演示如何使用XStream进行XML与Java对象的转换: ```java import com.thoughtworks.xstream.XStream; class User { private String name; private int age; // getters and setters ...

    xStream 实现xml与 java对象的转换

    3. **初始化xStream**:创建xStream实例并进行配置,以指定如何将Java对象转换为XML。 ```java XStream xstream = new XStream(); xstream.alias("person", Person.class); // "person"是XML中的标签名 ``` 4. ...

    XML转换为JAVA对象的方法

    在提供的`AppDemo`文件中,可能包含了一个简单的示例应用,演示了如何在实际项目中使用XStream进行XML和Java对象的转换。通过运行和分析这个例子,你可以更好地理解这一过程,并将其应用到自己的项目中。 总之,...

    com.thoughtworks.xstream.XStream操作XML和java对象的一些用法

    本篇将详细介绍XStream如何处理XML与Java对象的相互转换,以及如何处理List、Map等集合类型的数据。 首先,XStream是一个强大的库,它由Pete Gyford开发,用于将Java对象序列化为XML,同时也能将XML反序列化为Java...

    xStream学习xml,JASON,JAVA对象转换

    这篇博客文章,标题为“xStream学习xml,JASON,JAVA对象转换”,显然会探讨如何利用XStream来处理这两种数据格式。 XStream的核心功能在于其简单易用的API,可以将Java对象直接序列化成XML或JSON字符串,同时也能...

    XStream测试Demo-xml与java对象的相互转换

    首先,创建一个XStream实例,然后使用`toXML()`方法将Java对象转换为XML字符串: ```java XStream xstream = new XStream(); User user = new User(); user.setName("张三"); user.setAge(30); String xml = ...

    XStream在Java对象和XML之间相互转换

    - 配置文件:对于需要复杂结构的配置文件,XStream可以将配置对象直接转换为XML,简化配置管理。 6. **XStream的特性**: - 自动映射:XStream会自动将Java类的字段映射到XML元素或属性,无需手动编写XML Schema...

    Java用xStream拼接xml

    xStream的核心功能是基于对象关系映射(ORM)的概念,它自动将Java类的实例转换为XML文档,并能将XML文档转换回相应的Java对象。 **xStream的使用** 1. **添加依赖** 在使用xStream之前,你需要将其作为项目依赖...

    xstream对象与xml转换

    反序列化即把XML转换回Java对象。首先,你需要确保XML字符串是有效的,然后使用`fromXML()`方法: ```java String xmlInput = ... // 假设这是上面生成的XML Person deserializedPerson = (Person) xstream.fromXML...

    利用XStream读写xml文件

    为了能够通过XStream进行XML与Java对象之间的转换,首先需要定义相应的实体类。示例代码中定义了三个实体类: - **Company**:表示公司,包含多个部门。 - 属性:`List&lt;Department&gt; departments` - 方法:`...

    使用XStream是实现XML与Java对象的转换(5)--Object Stream

    首先创建XStream实例,然后使用`toXML()`方法将Java对象转换为XML字符串: ```java XStream xstream = new XStream(); User user = new User(); user.setName("Forest"); user.setAge(30); String xml = xstream.to...

    好用的xStream XML和JAVA对象互相转换的简单示例

    XStream是一个轻量级、高性能的库,能够将Java对象序列化为XML,同时也能将XML反序列化回Java对象,实现XML和Java对象的互相转换。 首先,我们需要引入XStream库。在给定的描述中提到了两个依赖文件:xstream-1.3.1...

    XStream解析xml 转换成对象案例

    而XStream是一个Java库,它能够将Java对象序列化为XML,反之也能将XML反序列化回Java对象,极大地简化了Java与XML之间的交互。本篇文章将深入探讨如何使用XStream进行XML到对象的转换,并提供一个实际的案例。 首先...

    Xstream_java对象和xml的互相转换

    XStream 是一个强大的 Java 库,它允许程序员轻松地将 Java 对象序列化为 XML,同时也能从 XML 数据中反序列化回原来的 Java 对象。这个功能在数据持久化、网络传输或者配置文件存储等方面非常有用。XStream 的设计...

    xml解析及使用XStream实现javaBean与xml之间的转换

    在`xmlAnalysis`文件夹中,可能包含了一个简单的Java程序,演示了如何使用DOM、SAX、StAX和XStream解析XML文件,并展示了XStream如何在JavaBean与XML之间进行转换。你可以运行这些代码,观察输出结果,以加深理解。...

    xstream 1.3.1.jar bean对象转换为xml

    在这个版本中,XStream提供了对Java Bean对象转换为XML的全面支持。 **XStream的工作原理:** XStream的核心是基于反射的,它通过分析Java对象的类结构来构建XML表示。它使用了一种称为"alias"的概念,允许我们将...

    android XMl 解析神奇xstream 二: 把对象转换成xml

    XStream是一个强大的库,它使得Java对象与XML之间的转换变得极其简单。本篇文章将深入探讨如何在Android环境中利用XStream将对象转换成XML。 首先,我们需要理解XStream的基本工作原理。XStream的核心功能是它可以...

    使用XStream是实现XML与Java对象的转换(3)--注解

    首先,XStream库的核心理念是通过简单的映射机制将Java对象直接转换为XML,反之亦然。通过注解,我们可以更加灵活地控制对象到XML的映射规则,避免编写大量的配置代码。在XStream中,主要使用以下几种注解: 1. `@...

    xStream转换xml和json源码

    xStream是一个轻量级的Java库,用于将Java对象序列化为XML,同时也能将XML反序列化回Java对象。这个强大的工具同样支持JSON格式,使得在Java应用程序中处理XML和JSON数据变得非常便捷。本篇文章将深入探讨xStream...

Global site tag (gtag.js) - Google Analytics