JAXB:序列化嵌套映射时获取空值

时间:2013-10-18 14:37:11

标签: java xml jaxb

我正在尝试使用JAXB来序列化XML中的类。

@XmlRootElement
class Foo
{
   Hashtable<String, Hashtable<String, Integer>> table = new Hashtable<>();

   public Hashtable<String, Hashtable<String, Integer>> getTable() {
     return table;
   }

   public void setTable(Hashtable<String, Hashtable<String, Integer>> t) {
     table = t;
  }
}

但是,这会产生具有空值的XML(我向您保证这些值实际存在!)

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<Foo>
    <table>
        <entry>
            <key>Test key</key>
            <value/>
        </entry>
    </table>
</Foo>

有没有一种简单的方法来解决这个问题?除非我真的需要,否则我真的不想使用@XmlJavaTypeAdapters

使用普通哈希表工作正常:Hashtable<String, Integer>> table = new Hashtable<>();

1 个答案:

答案 0 :(得分:1)

很抱歉让您失望,但目前没有简单的方法可以解决此问题。外部和内部哈希表之间存在显着差异。外部属性是属性,由com.sun.xml.bind.v2.runtime.property.SingleMapNodeProperty<BeanT, ValueT>类在内部处理。这个类可以将地图表示为键/值条目。

然而,在内部Hashtable的情况下,它不是“静态”属性,而是动态属性。这就是它由通用com.sun.xml.bind.v2.model.impl.RuntimeClassInfoImpl处理的原因。该类在Hashtable中找不到任何JAXB属性(即java bean属性 - 同时包含getter和setter)。因此,您将获得空value元素。

由于同样的原因,以下动态Hashtable属性也不起作用:

@XmlRootElement
@XmlSeeAlso(Hashtable.class)
public static class TypeWithHashtableAsObject {
    private Object property;

    public Object getProperty() {
        return property;
    }

    public void setProperty(Object property) {
        this.property = property;
    }

}

...
TypeWithHashtableAsObject foo = new TypeWithHashtableAsObject();
Hashtable<String, Integer> property = new Hashtable<>();
property.put("innerKey", 12);
foo.setProperty(property);
StringWriter writer = new StringWriter();
marshaller.marshal(foo, writer);
System.out.println(writer.toString());

结果:

<typeWithHashtableAsObject>
    <property xsi:type="hashtable" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"/>
</typeWithHashtableAsObject>

那是空元素。

In another answer您可以找到更多如何编组嵌套集合的示例。另一种解决方案是用另一种类型包装Hashtable。像表:

public class Table {
    private Hashtable<String, Integer> table = new Hashtable<>();

    public Table(Hashtable<String, Integer> table) {
        this.table = table;
    }

    public Table() {

    }

    public Hashtable<String, Integer> getTable() {
        return table;
    }

    public void setTable(Hashtable<String, Integer> table) {
        this.table = table;
    }

}

并将Foo.table类型更改为Hashtable<String, Table>

结果比原始结果更冗长,但恕我直言非常一致:

<foo>
   <table>
      <entry>
         <key>key1</key>
         <value>
            <table>
               <entry>
                  <key>innerKey</key>
                  <value>12</value>
               </entry>
            </table>
         </value>
      </entry>
   </table>
</foo>