#java #xml #xstream
#java #xml #xstream
Вопрос:
Я пытаюсь десериализовать такой XML-документ:
<rootelem>
<elementType1 arg1="..." />
<elementType1 arg1="..." />
<elementType1 arg1="..." />
<elementType2 argA="..." argB="..." />
<elementType2 argA="..." argB="..." />
<elementType2 argA="..." argB="..." />
</rootelem>
По умолчанию XStream может анализировать только такую форму:
<rootelem>
<list1>
<elementType1 arg1="..." />
<elementType1 arg1="..." />
<elementType1 arg1="..." />
</list1>
<list2>
<elementType2 argA="..." argB="..." />
<elementType2 argA="..." argB="..." />
<elementType2 argA="..." argB="..." />
</list>
</rootelem>
Это потому, что XStream использует следующий формат для коллекций:
<collection>
<elem .... />
<elem .... />
<elem .... />
</collection>
и теги фрейма обязательны. Коллекция может содержать только узлы одного типа. Итак, как я могу проанализировать такой XML-документ? Теперь я написал свой собственный конвертер для этого, но мне интересно, есть ли какие-то другие способы?
Ответ №1:
Я думаю, что неявные коллекции — это решение для вас.
http://x-stream.github.io/alias-tutorial.html#implicit
Вот пример кода:
package com.thoughtworks.xstream;
import java.util.ArrayList;
import java.util.List;
public class Test {
public static void main(String[] args) {
Blog teamBlog = new Blog(new Author("Guilherme Silveira"));
teamBlog.add(new Entry("first","My first blog entry."));
teamBlog.add(new Entry("tutorial",
"Today we have developed a nice alias tutorial. Tell your friends! NOW!"));
XStream xstream = new XStream();
xstream.alias("blog", Blog.class);
xstream.alias("entry", Entry.class);
xstream.addImplicitCollection(Blog.class, "entries");
System.out.println(xstream.toXML(teamBlog));
}
}
И результат:
<blog>
<author>
<name>Guilherme Silveira</name>
</author>
<entry>
<title>first</title>
<description>My first blog entry.</description>
</entry>
<entry>
<title>tutorial</title>
<description>
Today we have developed a nice alias tutorial. Tell your friends! NOW!
</description>
</entry>
</blog>
Ответ №2:
Первое, что вам нужно сделать, это создать POJO для всех ваших основных XML-тегов. Я предполагаю, что приведенный вами пример не является фактическим XML, который вам нужно «разархивировать» (это XML-жаргон для десериализации), но я воспользуюсь им, чтобы дать вам пример для работы ;-).
public class ElementType1
{
private String arg1;
public ElementType1()
{
setArg1("");
}
public String getArg1()
{
return arg1;
}
public void setArg1(String newArg1)
{
arg1 = newArg1;
}
}
public class ElementType2
{
private String argA;
private String argB;
public ElementType2()
{
setArgA("");
setArgB("");
}
public String getArgA()
{
return argA;
}
public void setArgA(String newArgA)
{
argA = newArgA;
}
public String getArgB()
{
return argB;
}
public void setArgB(String newArgB)
{
argB = newArgB;
}
}
public class RootElement
{
private List<ElementType1> element1s;
private List<ElementType2> element2s;
public RootElement()
{
setElement1s(new ArrayList<ElementType1>());
setElement2s(new ArrayList<ElementType2>());
}
public List<ElementType1> getElement1s()
{
return element1s;
}
public void setElement1s(List<ElementType1> newElement1s)
{
element1s = newElement1s;
}
public List<ElementType2> getElement2s()
{
return element2s;
}
public void setElement2s(List<ElementType2> newElement2s)
{
element2s = newElement2s;
}
}
Теперь, когда у вас есть ваши POJOS, использовать XStream для маршаллинга (сериализации) и отмены маршаллинга (десериализации) их в XML и из XML довольно просто.
List<ElementType1> e1 = new ArrayList<ElementType1>();
List<ElementType2> e2 = new ArrayList<ElementType2>();
ElementType1 a, b, c;
a = new ElementType1();
b = new ElementType1();
c = new ElementType1();
a.setArg1("fizz");
b.setArg1("buzz");
c.setArg1("foo");
e1.add(a);
e1.add(b);
e1.add(c);
ElementType2 d, e;
d = new ElementType2();
e = new ElementType2();
d.setArgA("flim");
d.setArgB("flam");
e.setArgA("something");
e.setArgB("blah");
e2.add(d);
e2.add(e);
RootElement rootElem = new RootElement();
rootElem.setElement1s(e1);
rootElem.setElement2s(e2);
XStream xstream = new XStream();
RootElement rootElem = getYourRootElementSomehow();
String rootElementAsXml = xstream.toXML(rootElem);
System.out.println(rootElementAsXml);
Теперь этот код выведет на консоль следующее:
<fully.qualified.pkg.name.RootElement>
<element1s>
<fully.qualified.pkg.name.ElementType1>
<arg1>fizz</arg1>
</fully.qualified.pkg.name.ElementType1>
<fully.qualified.pkg.name.ElementType1>
<arg1>buzz</arg1>
</fully.qualified.pkg.name.ElementType1>
<fully.qualified.pkg.name.ElementType1>
<arg1>foo</arg1>
</fully.qualified.pkg.name.ElementType1>
</element1s>
<element2s>
<fully.qualified.pkg.name.ElementType2>
<argA>flim</argA>
<argB>flam</argB>
</fully.qualified.pkg.name.ElementType2>
<fully.qualified.pkg.name.ElementType2>
<argA>something</argA>
<argB>blah</argB>
</fully.qualified.pkg.name.ElementType2>
</element2s>
</fully.qualified.pkg.name.RootElement>
Затем вы можете очистить неприятный, полнофункциональный пакет с именем XML elements, используя универсальные методы «псевдонимов» XStreams, например, так:
xstream.alias("elementType1", ElementType1.class);
xstream.alias("elementType2", ElementType2.class);
xstream.alias("rootElement", RootElement.class);
String rootElementAsXml = xstream.toXML(rootElem);
System.out.println(rootElementAsXml);
Который теперь будет распечатан:
<rootElement>
<element1s>
<elementType1>
<arg1>fizz</arg1>
</elementType1>
<elementType1>
<arg1>buzz</arg1>
</elementType1>
<elementType1>
<arg1>foo</arg1>
</elementType1>
</element1s>
<element2s>
<elementType2>
<argA>flim</argA>
<argB>flam</argB>
</elementType2>
<elementType2>
<argA>something</argA>
<argB>blah</argB>
</elementType2>
</element2s>
</rootElement>
Теперь я знаю, что в вашем примере вы хотели, чтобы все <arg>
s на самом деле были атрибутами <elementType>
тегов, а не сами дочерние теги. Вы можете использовать мощный API XStream, чтобы сделать это за вас, как вам нравится. Но, надеюсь, этого достаточно, чтобы вы начали.
И, кстати, превратить ваш XML обратно в POJO так же просто:
RootElement root = (RootElement)xstream.fromXML(rootElementAsXml);
Как правило, XStream — это «oxmapper» (объектно-XML-сопоставитель), и он поставляется предварительно созданным, чтобы знать, как превратить ваш POJO в значимый XML и наоборот. Вы всегда можете переопределить его значения по умолчанию, но чаще всего нет. Я всегда удивлялся тому, насколько умны его значения по умолчанию. Желаю вам удачи.
Комментарии:
1. Спасибо вам за столь подробный ответ! Но я уже знаю об этом, мой главный вопрос — как мы можем опустить теги фрейма списка:
<list a="1"> <listitem param = ""/>... </list>
вместо<list a="1"> <arraylist><listitem param = "" />...</arraylist></list>
2. Извините, все еще не уверен, что вы ищете. Можете ли вы указать точный XML, который вы хотите (либо в комментарии, либо отредактировав свой исходный вопрос), используя ваши теги <rootElement> и <ElementType>? Если я увижу пример именно того, что вы ищете, возможно, мы сможем вам помочь.
3. Аааа… Я понимаю. Извините, Lampapos, насколько мне известно, то, что вам нужно, невозможно в XStream. Во всех классах com.thoughtworks.xstream.converters.collections.* (которые XStream использует для OXM с java.util. Коллекции), в документации четко указано, что каждому элементу в коллекции присваивается свой собственный вложенный элемент. Я не верю, что это возможно переопределить. Мои серьезные предложения: либо (а) измените свою схему, (б) преобразуйте свою схему непосредственно перед ее использованием в XStream, либо (в) вам придется смириться с этим. Извините, у меня нет лучшего прогноза для вас.