Jaxb2Marshaller无法以线程安全的方式解组非XmlRoot元素

时间:2018-02-05 17:01:09

标签: java xml spring spring-boot jaxb

我有一堆从架构生成的类。我无法触摸它们 - 对我来说是不可改变的来源。

这些类是架构中的顶级元素。每个都指向一个相同名称的类型:

    <xsd:element name="Foo" type="FooType"/>

生成一个具有所有必要属性的FooType类,但它没有@XmlRoot注释。未生成课程Foo

在JAXB中,实现使用查找表并明确地解组该对象:

JAXBContext ctx ...
FooType x = ctx.createUnmarshaller().unmarshal(source, FooType.class);

在spring-boot中,使用Jaxb2Marshaller,我无法给出类参数。在那里看来,编组只依赖@XmlRoot注释来做它的魔力。 设置marshaller.setMappedClass(FooType.class)时,springboot unmarshaller将使用它 - 但是,这只在实例中设置一次而不是线程安全,只有在我按需创建Jaxb2Marshaller时(看起来很昂贵)。

我需要访问JAXBContext本身,以创建我自己的unmarshaller。虽然这是可能的,但我没有在unmarshaller上得到神奇的初始化。有用的方法Jaxb2Marshaller#createUnmarshaller()protected

我只看到一个解决方案,将Jaxb2Marshaller扩展到我自己的虚拟对象,以扩大对受保护方法的访问。这似乎是不洁净的。

我宁愿完全删除Jaxb2Marshaller并继续使用遗留应用程序并直接使用JAXB。 Spring的JAXB似乎不打算在这里介绍我的情况。

问题:我是否错过理解这个概念? Jaxb2Marshaller.setMappedClass()的用途是什么?

1 个答案:

答案 0 :(得分:0)

也许我没有完全解决您问题的核心问题,但如果缺少@XmlRootElement是您的问题,我建议您稍微调整您的架构。

我认为你有类似的东西:

<xsd:element name="Foo" type="FooType"/>

<!-- and somewhere in the schema also have -->
<xsd:complexType name="FooType">
    <xsd:sequence>
        <xsd:element name="blah" type="BlahType"/>
        <xsd:element name="blah-blah" type="BlahBlahType" />
    </xsd:sequence>
</xsd:complexType>

您需要做的是删除上面指定的两个并替换为:

<xsd:element name="Foo">
    <xsd:complexType>
        <xsd:sequence>
            <xsd:element name="blah" type="BlahType"/>
            <xsd:element name="blah-blah" type="BlahBlahType" />
        </xsd:sequence>
    </xsd:complexType>
</xsd:element>

当您将与Foo一起生成类时,它将自动拥有@RootXmlElement