我有一个相当复杂的设置。 我有一个超级项目(所有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中。
说到这一点,我想要做的是编写XML,以验证XSD的这种设置,但是我没有这样做。 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也会)抱怨:
元素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标签。可以这样做吗?
似乎我在2019年与JAXB是一个孤独的战斗机。然而,在主要通过写这个问题与橡皮鸭交谈之后,我尝试了一些更多的组合,解决方案是这样的:
超级项目和子项目都必须位于同一名称空间中。我起初没有采用这种解决方案,因为我认为我无法单独验证子项目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,这正是我需要的。