如何编写具有多个名称空间的正确XML以便进行验证?

时间:2019-02-14 15:36:59

标签: java xml xsd jaxb

我的设置比较复杂。
我有一个超级项目(所有Maven)和两个子项目。这两个子项目无关。我正在设计所有项目中的类,并使用JAXB批注对其进行批注,以便在每个子项目级别上创建XSD。每个子项目都应该包含超级项目中的XML类。无需在超级项目级别使用XSD。

在超级项目中,我有几个软件包,每个软件包都有一个package-info.java文件,如下所示:

@javax.xml.bind.annotation.XmlSchema(elementFormDefault = javax.xml.bind.annotation.XmlNsForm.QUALIFIED, xmlns = {
    @javax.xml.bind.annotation.XmlNs(prefix = "projectA", namespaceURI = "https://some.url.to/projectA"),
    @javax.xml.bind.annotation.XmlNs(prefix = "projectB", namespaceURI = "https://some.url.to/projectB")
})

为简单起见,我们将其称为子项目projectA和projectB。
如前所述,在子项目中,我还有几个带有package-info.java文件的软件包,如下所示:

@javax.xml.bind.annotation.XmlSchema(namespace = "https://some.url.to/projectA", elementFormDefault = javax.xml.bind.annotation.XmlNsForm.QUALIFIED)

在项目(A | B)中,我正在从此设置生成XSD架构。这样可以正常工作,并且找到并解决了从子项目到超级项目的所有引用。

JAXB为每个子项目生成两个XSD文件。在下面的示例中,让我们坚持项目A。然后,schema1.xsd必须具有以下标头:

<xs:schema elementFormDefault="qualified" version="1.0" targetNamespace="https://some.url.to/projectA" xmlns:tns="https://some.url.to/projectA" xmlns:xs="http://www.w3.org/2001/XMLSchema">

之后

<xs:import schemaLocation="schema2.xsd"/>

schema2.xsd具有以下标头:

<xs:schema elementFormDefault="qualified" version="1.0" xmlns:projectB="https://some.url.to/projectB" xmlns:projectA="https://some.url.to/projectA" xmlns:xs="http://www.w3.org/2001/XMLSchema">

请注意,出于某种原因,即使在XSD中未引用projectB,也包含在此处。在项目B中是另一回事。

在schema1.xsd中,我发现像这样的行:

<xs:element name="a_name" type="a_type_in_schema2"/>
<xs:element name="another_name" type="tns:a_type_in_schema1" use="required"/>

在schema2.xsd中,我发现像这样的行:

<xs:element name="something" type="another_type_in_schema2" form="unqualified"/>

请注意“不合格”属性。它也会出现在schema2.xsd中定义的复杂类型的每个元素上。

这意味着,超级项目中的所有内容都包含在schema2.xsd中,而仅项目A中定义的所有内容都包含在schema1.xsd中。

到目前为止,我想做的是编写针对此XSD设置进行验证的XML,但是我做不到。
XML的标头是:

<rootNode xmlns="https://some.url.to/projectA" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="https://some.url.to/projectA https://some.url.to/projectA.xsd">

在这种XML中,每当我尝试引用在schema2.xsd(=超级项目中的类型)中定义的类型时,

<a_type_in_schema1>
    <a_type_in_schema2 .... />
</a_type_in_schema1>

xmllint(还有Eclipse)也会抱怨:

  

element a_type_in_schema2:模式有效性错误:元素   '{https://some.url.to/projectA} a_type_in_schema2':该元素不是   预期。预期是(a,b,c,a_type_in_schema2,d,e)之一。

我怎么可能正确地编写它?

请注意,我不希望使用<namespace:a_type_in_schema2>之类的限定标签。由于可用性,我只希望使用“普通” XML标签。能做到吗?

1 个答案:

答案 0 :(得分:1)

似乎我在2019年是JAXB的一名孤单战士。但是,在主要通过写这个问题与橡皮鸭交谈之后,我尝试了更多组合,而解决方案是:

super和sub项目都必须在同一个命名空间中。一开始我并没有选择这种解决方案,因为我认为我将无法单独验证子项目XSD。但这是可能的。

所以,在超级项目中,我只添加了

@javax.xml.bind.annotation.XmlSchema(namespace = "https://some.url.to/commonNS", elementFormDefault = javax.xml.bind.annotation.XmlNsForm.QUALIFIED)

两个子项目中使用了相同的架构定义。

然后,我可以在XML标头中编写

<rootNode xmlns="https://some.url.to/commonNS" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="https://some.url.to/commonNS https://some.url.to/projectA.xsd">

请注意xsi_schemaLocation中的重要更改。我可以在这里使用特定的XSD验证commonNS命名空间。在项目B中,我将使用对其XSD的引用。

所以我有一个命名空间,但是有不同的XSD文件来验证给定的XML,这正是我所需要的。