XStream несколько типов узлов в коллекции signle

#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, либо (в) вам придется смириться с этим. Извините, у меня нет лучшего прогноза для вас.