JAXB картографиращи елементи с неизвестно име

Имам XML, който е извън моя контрол върху това как се генерира. Искам да създам обект от него, като го демаршалирам в клас, написан на ръка от мен.

Един фрагмент от неговата структура изглежда така:

<categories>
    <key_0>aaa</key_0>
    <key_1>bbb</key_1>
    <key_2>ccc</key_2>
</categories>

Как мога да се справя с такива случаи? Разбира се, броят на елементите е променлив.


person bbcooper    schedule 25.11.2010    source източник
comment
XML структурата има ли добре дефинирана схема?   -  person Vineet Reynolds    schedule 25.11.2010
comment
Не. Източникът е някаква PHP REST уеб услуга (което не означава, че не може да се направи).   -  person bbcooper    schedule 25.11.2010


Отговори (3)


Ако използвате следния обектен модел, тогава всеки от некартираните елементи key_# ще се запази като екземпляр на org.w3c.dom.Element:

import java.util.List;
import javax.xml.bind.annotation.XmlAnyElement;
import javax.xml.bind.annotation.XmlRootElement;
import org.w3c.dom.Element;

@XmlRootElement
public class Categories {

    private List<Element> keys;

    @XmlAnyElement
    public List<Element> getKeys() {
        return keys;
    }

    public void setKeys(List<Element> keys) {
        this.keys = keys;
    }

}

Ако някой от елементите съответства на класове, съпоставени с анотация @XmlRootElement, тогава можете да използвате @XmlAnyElement(lax=true) и известните елементи ще бъдат преобразувани в съответните обекти. За пример вижте:

person bdoughan    schedule 25.11.2010
comment
Да, това е решението. Модифицирах го малко (премахнах мутаторите) и го включих като вътрешен статичен клас. Страхотно, сега преминаваме към следващите проблеми... - person bbcooper; 26.11.2010

Използвайте така

        @XmlRootElement
        @XmlAccessorType(XmlAccessType.FIELD)
        public static class Categories {


            @XmlAnyElement
            @XmlJavaTypeAdapter(ValueAdapter.class)
            protected List<String> categories=new ArrayList<String>();

            public List<String> getCategories() {
                return categories;
            }
            public void setCategories(String value) {
                this.categories.add(value);
            }
        }

        class ValueAdapter extends XmlAdapter<Object, String>{

           @Override
           public Object marshal(String v) throws Exception {
              // write code for marshall
             return null;
           }

           @Override
           public String unmarshal(Object v) throws Exception {
              Element element = (Element) v;
              return element.getTextContent();
          }
       }
person ysramkumar    schedule 22.06.2015

За този прост елемент бих създал клас, наречен Categories:

import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement
public class Categories {

    protected String key_0;
    protected String key_1;
    protected String key_2;

    public String getKey_0() {
        return key_0;
    }

    public void setKey_0(String key_0) {
        this.key_0 = key_0;
    }

    public String getKey_1() {
        return key_1;
    }

    public void setKey_1(String key_1) {
        this.key_1 = key_1;
    }

    public String getKey_2() {
        return key_2;
    }

    public void setKey_2(String key_2) {
        this.key_2 = key_2;
    }

}

След това в основен метод или нещо подобно, бих създал unmarshaller:

JAXBContext context = JAXBContext.newInstance(Categories.class);
Unmarshaller um = context.createUnmarshaller();
Categories response = (Categories) um.unmarshal(new FileReader("my.xml"));
// access the Categories object "response"

За да мога да извлека всички обекти, мисля, че бих поставил всички елементи вътре в основен елемент в нов xml файл и ще напиша клас за този основен елемент с анотацията @XmlRootElement.

Надявам се, че това помага, човече

person mman    schedule 25.11.2010
comment
+1, но не е необходимо да посочвате име в XmlRootElement, тъй като то ще се използва по подразбиране за категории, също така не е необходимо да коментирате с @XmlElement, тъй като е по подразбиране. - person bdoughan; 25.11.2010
comment
не Знам как да маршалирам и демаршалирам прости класове. Може би не бях достатъчно ясен във въпроса си. Горният XML е САМО ФРАГМЕНТ от по-голям XML. Въпросът ми беше как да се справя с НЕИЗВЕСТЕН ​​брой елементи под елемента ‹category›. Може да има ‹key_67›, следователно 67 елемента под елемента ‹category›. - person bbcooper; 25.11.2010
comment
Винаги ли съдържанието на елемента key_# е само текстов възел? - person bdoughan; 25.11.2010
comment
Да, винаги е текстов възел. - person bbcooper; 25.11.2010