JAXB删除不必要的嵌套XML标记

时间:2014-08-11 15:06:56

标签: java xml java-ee jaxb javax.xml

目前我的节目有以下输出:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<container>
    <elements>
        <property name="e1">
            <foo name="Alex" status="Married"/>
        </property>
        <property name="e2">
            <foo name="Chris" status="Married with 2 children"/>
        </property>
    </elements>
</container>

正如您所看到的,同时拥有<container><elements>标签是没用的。我想删除<elements>,因此输出结果如下:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<container>
    <property name="e1">
        <foo name="Alex" status="Married"/>
    </property>
    <property name="e2">
        <foo name="Chris" status="Married with 2 children"/>
    </property>
</container>

下面列出了生成第一个输出的代码:

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Container {

    @XmlElement
    @XmlJavaTypeAdapter(MyAdapter.class)
    private Map<String, Foo> elements = new HashMap<String, Foo>();

    public Container() {
        this.elements = new HashMap<String, Foo>();
    }

    public Map<String, Foo> getElements() {
        return elements;
    }

    @XmlAccessorOrder(XmlAccessOrder.ALPHABETICAL)
    @XmlRootElement(name = "foo")
    static class Foo {
        @XmlAttribute
        public String name;

        @XmlAttribute
        public String status;

        public Foo(String name, String status) {
            this.name = name;
            this.status = status;
        }

        public Foo() {
        }
    }

    public static void main(String[] args) throws JAXBException {
        final JAXBContext context = JAXBContext.newInstance(Container.class);
        final Marshaller m = context.createMarshaller();
        m.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);

        final Container c = new Container();
        final Map<String, Foo> elementsMap = c.getElements();
        elementsMap.put("e1", new Foo("Alex", "Married"));
        elementsMap.put("e2", new Foo("Chris", "Married with 2 children"));

        m.marshal(c, System.out);
    }
}

MyAdapter课程,基于JAXB @XmlAdapter: Map -> List adapter? (marshall only)

public class MyAdapter extends XmlAdapter<MyAdapter.AdaptedFoo, Map<String, Foo>> {

    static class AdaptedFoo {
        public List<Property> property = new ArrayList<>();
    }

    public static class Property {
        @XmlAttribute
        public String name;

        @XmlElementRef(type = Foo.class)
        public Foo value;
    }

    @Override
    public Map<String, Foo> unmarshal(AdaptedFoo v) throws Exception {
        return null;
    }

    @Override
    public AdaptedFoo marshal(Map<String, Foo> map) throws Exception {
        if (null == map) {
            return null;
        }
        AdaptedFoo adaptedFoo = new AdaptedFoo();
        for (Entry<String, Foo> entry : map.entrySet()) {
            Property property = new Property();
            property.name = entry.getKey();
            property.value = entry.getValue();
            adaptedFoo.property.add(property);
        }
        return adaptedFoo;
    }
}

如何从输出中删除<elements>代码?


编辑:我发现了“肮脏的”&#39;这样做的方法 - 为@XmlRootElement(name = "")设置Container。但是有没有优雅的&#39;方式是什么?

2 个答案:

答案 0 :(得分:1)

需要XML元素<elements>将封闭元素序列与属性Map<?,?> elements相关联。你不能放弃它:unmarshaller如何在元素<property>的层面上知道<container>元素属于哪个*。

由于JAXB将重复的元素x“硬连线”处理为List<Property>,因此List<?> x是不同的,因此不需要包装器。

由于您编写带注释的Java类,您可以通过添加(到Container)另一个“虚拟”字段并修改一些注释来使用它:

@XmlAccessorType(XmlAccessType.PROPERTY)
public class Container {
// ...
@XmlTransient    
public Map<String,Foo> getElements(){
    return elements;
}
private List<Property> property;
@XMLElement
public List<Property> getProperty(){
    List<Property>  props = elements.entrySet().stream()
                            .map( e -> new Property( e.getKey(), e.getValue() )
                            .collect( Collectors.toList() );
    return props;
}

较旧的Javas可能会

   List<Property> props = new ArrayList<>();
   for( Map.Entry<String,Foo> e: elements.entrySet() ){
       props.add( new Property( e.getKey(), e.getValue() ) );
   }
   return props;

这样的警察:

<container>
    <property name="aaa">
       <foo name="aaaname" status="aaastatus"/>
    </property>
    <property name="bbb">
       <foo name="bbbname" status="bbbstatus"/>
    </property>
</container>

它没有解组!

答案 1 :(得分:0)

除了通过&#34; hacking&#34;删除标签。元素名称为&#34;&#34;,我不认为你能够在没有嵌套标签的情况下使用该数据类型。

标记描述了包含您的自适应类的数据类型,您已将其注释为名为elements的XML元素,其中包含由适配器定义的属性。

如果你想要的东西更符合你的想法,可以改变你的根类的上下文,或者在容器类中使用List而不是HashMap。< / p>