XSD对特定类型的兄弟元素属性的唯一约束

时间:2012-04-27 19:29:57

标签: xml xpath xsd unique

我有一个结构为Q& A的XML文档,它遵循以下格式(为清晰起见而编辑):

<question>
    <answer id="1">
        <question>
            <answer id="1"/>
            <answer id="2"/>
            <answer id="3"/>
        </question>
    </answer>
    <answer id="2">
        <question>
            <answer id="1"/>
            <answer id="2"/>
        </question>
    </answer>
</question>

我的XSD看起来像这样:

<xs:element name="question">
     <xs:complexType>
        <xs:sequence>
            <xs:element name="answer" type="answerType" minOccurs="2" maxOccurs="unbounded">
            </xs:element>
        </xs:sequence>
    </xs:complexType>
    <xs:unique name="AnswerIdUnique">
        <xs:selector xpath="./*" />
        <xs:field xpath="@id" />
    </xs:unique>
</xs:element>

<xs:complexType name="answerType">
    <xs:sequence>
        <xs:element ref="question" minOccurs="0" maxOccurs="1" />
    </xs:sequence>
    <xs:attribute name="id" type="xs:token" use="required" />
</xs:complexType>

当然,它比你上面看到的更多,但这说明了我的问题。 我需要id元素上的answer属性在兄弟姐妹中是唯一的。上面定义的XSD强制兄弟元素之间的id属性的唯一性,但它不区分元素类型。我在独特约束中尝试了各种选择器和字段,但没有找到有效的组合。 有什么建议吗?

1 个答案:

答案 0 :(得分:14)

只需将选择器更改为<xs:selector xpath="answer"/>,您就可以了。一般情况下,如果仅出于性能原因,最好避免使用.//*之类的XPath。

这是您提供的XML示例的XML Schema,我认为它正如您所希望的那样工作:

<?xml version="1.0" encoding="utf-8" ?>
<xs:schema elementFormDefault="qualified" xmlns:xs="http://www.w3.org/2001/XMLSchema">
    <xs:element name="question" type="questionType">
        <xs:unique name="AnswerIdUnique">
            <xs:selector xpath="answer"/>
            <xs:field xpath="@id"/>
        </xs:unique>
    </xs:element>
    <xs:complexType name="questionType">
        <xs:sequence>
            <xs:element name="answer" type="answerType" minOccurs="2" maxOccurs="unbounded"/>
        </xs:sequence>
    </xs:complexType>
    <xs:complexType name="answerType">
        <xs:sequence>
            <xs:element ref="question" minOccurs="0" maxOccurs="1"/>
        </xs:sequence>
        <xs:attribute name="id" type="xs:token" use="required"/>
    </xs:complexType>
</xs:schema>

您发布的XML符合上述要求;复制任何兄弟答案的id会产生验证错误。例如,以下XML:

<question> 
    <answer id="1"> 
        <question> 
            <answer id="1"/> 
            <answer id="2"/> 
            <answer id="1"/> 
        </question> 
    </answer> 
    <answer id="1"> 
        <question> 
            <answer id="1"/> 
            <answer id="2"/> 
        </question> 
    </answer> 
</question> 

验证时(在QTAssistant中,应该类似于Visual Studio中的消息,因为它基于相同的技术),这些是错误:

Error occurred while loading [], line 6 position 5
There is a duplicate key sequence '1' for the 'AnswerIdUnique' key or unique identity constraint.
Error occurred while loading [], line 9 position 3
There is a duplicate key sequence '1' for the 'AnswerIdUnique' key or unique identity constraint.
Document1.xml is invalid.

以下是Visual Studio 2010的屏幕截图,显示了针对我发布的XSD的上述XML验证;虽然这些问题无意中被报告为警告,但它们仍被报道。

VS2010 showing unique constraint errors

我随机选择了一个在线验证器(http://xsdvalidation.utilities-online.info/)并验证了我发布的相同XML和XSD;错误报告为:

org.xml.sax.SAXParseException: Duplicate unique value [1] declared for identity constraint of element "question".org.xml.sax.SAXParseException: Duplicate unique value [1] declared for identity constraint of element "question".

您必须注意的一件事是当您拥有XSD的目标命名空间时;在这种情况下,需要为所有涉及的命名空间定义别名,并在选择器中使用它们。

更新:带名称空间的XSD:

<?xml version="1.0" encoding="utf-8"?>
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:tns="http://localhost" xmlns="http://localhost" targetNamespace="http://localhost" elementFormDefault="qualified" attributeFormDefault="unqualified">
    <xs:element name="question" type="questionType">
        <xs:unique name="AnswerIdUnique">
            <xs:selector xpath="tns:answer"/>
            <xs:field xpath="@id"/>
        </xs:unique>
    </xs:element>
    <xs:complexType name="questionType">
        <xs:sequence>
            <xs:element name="answer" type="answerType" minOccurs="2" maxOccurs="unbounded"/>
        </xs:sequence>
    </xs:complexType>
    <xs:complexType name="answerType">
        <xs:sequence>
            <xs:element ref="question" minOccurs="0" maxOccurs="1"/>
        </xs:sequence>
        <xs:attribute name="id" type="xs:token" use="required"/>
    </xs:complexType>
</xs:schema>

请注意tns前缀的介绍及其在选择器中的使用。