Динамические атрибуты XML с JAXB

#java #jackson #jaxb #deserialization #xml-deserialization

#java — язык #джексон #jaxb #десериализация #xml-десериализация #java

Вопрос:

У меня есть XML файл, который я пытаюсь отменить привязку к объекту Java, используя JAXB

 <root>
    <object att1="orgA" att2="orgA" id="6" name="">
        ...
    </object>
</root>
  

Java класс:

 @AllArgsConstructor
@NoArgsConstructor
@ToString
@Setter
@XmlRootElement(name="object")
public class ObjectElement implements Serializable {

@XmlAttribute
private int id;

@XmlAttribute
private String attr1;

@XmlAttribute
private String attr2;

@XmlAttribute
private String name;

}
  

Вопрос: атрибуты xml-элемента object могут динамически меняться. Таким образом, ключи атрибутов могут быть динамическими, поэтому нет никакого способа поместить их как XmlAttributes в класс. Есть ли способ определить какой-то HashMap который считывает все ключи object и соответствующие значения? Например, новый объект может иметь совершенно другие атрибуты, подобные этому;

 <object att5="some" att7="other" id="6" name="value">
    ...
</object>
  

Комментарии:

1. Я думаю, вам нужно использовать XmlAdapter . Это может вам помочь.

Ответ №1:

Вам нужно написать пользовательский адаптер, который расширяет javax.xml.bind.annotation.adapters.XmlAdapter класс. В вашем случае мы можем создать Map , который представляет все свойства. После прочтения свойств вы можете задать все POJO поля с помощью reflection on или вручную. Пользовательский десериализатор, который динамически считывает все атрибуты, может выглядеть следующим образом:

 class ItemXmlAdapter extends XmlAdapter<Object, Item> {

    @Override
    public Item unmarshal(Object v) {
        Element element = (Element) v;

        Map<String, String> properties = new HashMap<>();
        NamedNodeMap attributes = element.getAttributes();
        for (int i = attributes.getLength() - 1; i >= 0; i--) {
            Node node = attributes.item(i);
            properties.put(node.getNodeName(), node.getNodeValue());
        }

        Item item = new Item();
        item.setProperties(properties);

        return item;
    }

    @Override
    public Object marshal(Item v) throws Exception {
        return null; // Implement if needed
    }
}
  

Простой пример приложения, которое считывает XML и анализирует все атрибуты:

 import org.w3c.dom.Element;
import org.w3c.dom.NamedNodeMap;
import org.w3c.dom.Node;

import javax.xml.bind.JAXBContext;
import javax.xml.bind.Unmarshaller;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlRootElement;
import javax.xml.bind.annotation.adapters.XmlAdapter;
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;
import java.io.File;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

public class JaxbApp {

    public static void main(String[] args) throws Exception {
        File xmlFile = new File("./resource/test.xml").getAbsoluteFile();

        JAXBContext context = JAXBContext.newInstance(Root.class);

        Unmarshaller unmarshaller = context.createUnmarshaller();
        Root root = (Root) unmarshaller.unmarshal(xmlFile);

        System.out.println(root);
    }
}

@XmlRootElement(name = "root")
class Root {

    private List<Item> items = new ArrayList<>();

    @XmlElement(name = "object")
    public List<Item> getItems() {
        return items;
    }

    // getters, setters, toString
}

@XmlJavaTypeAdapter(ItemXmlAdapter.class)
class Item {

    private Map<String, String> properties;

    // getters, setters, toString
}
  

Для приведенных ниже XML :

 <root>
    <object att1="orgA" att2="orgA" id="6" name="N"/>
    <object att5="some" id="6" name="value"/>
</root>
  

С принтами:

 Items{items=[{name=N, id=6, att2=orgA, att1=orgA}, {name=value, att5=some, id=6}]}