我正在寻找一种将Java对象序列化为XML以供RESTful Web服务使用的方法。我没有XSD。
我看过以下内容:
JAXB - 在类上需要注释以及ObjectFactory类和/或jaxb.index文件时相当重的重量
简单 - 需要注释,但没有其他配置类/文件。不幸的是,它无法序列化。
XStream - 不需要注释等,但不支持泛型
有没有人有任何建议?
答案 0 :(得分:4)
import java.beans.XMLEncoder;
import java.beans.XMLDecoder;
import java.io.*;
public class XMLSerializer {
public static void write(Object f, String filename) throws Exception{
XMLEncoder encoder =
new XMLEncoder(
new BufferedOutputStream(
new FileOutputStream(filename)));
encoder.writeObject(f);
encoder.close();
}
public static Object read(String filename) throws Exception {
XMLDecoder decoder =
new XMLDecoder(new BufferedInputStream(
new FileInputStream(filename)));
Object o = (Object)decoder.readObject();
decoder.close();
return o;
}
}
答案 1 :(得分:4)
JAXB非常轻松,并且附带Java 6,而且占用空间小。
以一个注释的价格得到这个。
Order.java
package xxx;
import java.util.Date;
import java.util.List;
import javax.xml.bind.annotation.XmlRootElement;
@XmlRootElement
public class Order {
int custId;
Date orderDate;
String description;
List<Item> items;
public void setCustId(int custId) {
this.custId = custId;
}
public void setOrderDate(Date orderDate) {
this.orderDate = orderDate;
}
public void setDescription(String description) {
this.description = description;
}
public void setItems(List<Item> items) {
this.items = items;
}
public int getCustId() {
return custId;
}
public Date getOrderDate() {
return orderDate;
}
public String getDescription() {
return description;
}
public List<Item> getItems() {
return items;
}
public String toString() {
return "Order: " + custId + " - " + orderDate + " - " + description + " - " + items;
}
}
Item.java:
package xxx;
public class Item {
String name;
private int qty;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getQty() {
return qty;
}
public void setQty(int qty) {
this.qty = qty;
}
public String toString() {
return "Item:" + name + " - " + qty;
}
}
Test.java:
package xxx;
import java.io.StringReader;
import java.io.StringWriter;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import javax.xml.bind.JAXBContext;
import javax.xml.bind.Marshaller;
import javax.xml.bind.Unmarshaller;
public class Test {
public static void main(String args[]) throws Exception {
Order o = new Order();
o.setCustId(123);
o.setDescription("New order");
o.setOrderDate(new Date());
List<Item> items = new ArrayList<Item>();
Item i = new Item();
i.setName("Duck Soup");
i.setQty(10);
items.add(i);
i = new Item();
i.setName("Carrots");
i.setQty(4);
items.add(i);
o.setItems(items);
//Write it
JAXBContext ctx = JAXBContext.newInstance(Order.class);
Marshaller m = ctx.createMarshaller();
m.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
StringWriter sw = new StringWriter();
m.marshal(o, sw);
sw.close();
System.out.println(sw.toString());
// Read it back
JAXBContext readCtx = JAXBContext.newInstance(Order.class);
Unmarshaller um = readCtx.createUnmarshaller();
Order newOrder = (Order) um.unmarshal(new StringReader(sw.toString()));
System.out.println(newOrder);
}
}
结果:
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<order>
<custId>123</custId>
<description>New order</description>
<items>
<name>Duck Soup</name>
<qty>10</qty>
</items>
<items>
<name>Carrots</name>
<qty>4</qty>
</items>
<orderDate>2010-06-16T18:12:06.870-07:00</orderDate>
</order>
Order: 123 - Wed Jun 16 18:12:06 PDT 2010 - New order - [Item:Duck Soup - 10, Item:Carrots - 4]
答案 2 :(得分:1)
我的投票将是XStream。缺乏仿制药支持是为它提供的灵活性支付的一小部分代价。您还可以通过在序列化时序列化泛型类类型来轻松实现泛型,或者将其构建到域对象中。 E.g。
class Customer
{
List<Order> orders;
public List<Order> getOrders()
{
return orders;
}
}
在流中,元素类型表示每个对象的类型。当类型是抽象类型或接口时,该元素与实现类一起列出,除非已将该类型指定为该接口类型的默认值。 (例如,ArrayList作为静态类型List实例的默认值。)
泛型是java中的“玫瑰色眼镜” - 它们并没有真正改变你所看到的,只是你如何看待它。如果它们是通过泛型支持发送的,那么这些对象将通过线路发送完全相同。
答案 3 :(得分:0)
您可以尝试JLibs XMLDocument。它使用SAX创建XML,因此重量轻,可以完全控制。