XSD:按顺序排列单个可选元素

时间:2012-07-05 09:22:19

标签: xml xsd

我有以下格式的XML文件:

<?xml version="1.0" encoding="ISO-8859-1" standalone="yes"?>
<hw:config name="test" xmlns:hw="hw">
  <switch var="var.test" >
    <case value="a">
    </case>
    <case value="b">
    </case>
    <default>
    </default>
    <case value="c">
    </case>
  </switch>
</hw:config>

我想要一个XSD,允许可能有一个默认块,但不能超过一个。它可能出现在案例块之间的任何位置。

以下XSD应该完成这项工作,但xmllint(libxml2)说“内容模型不是决定论者”。:

<?xml version="1.0"?>
<xsd:schema xmlns:xsd="http://www.w3.org/2001/XMLSchema"
      xmlns:hw="hw" targetNamespace="hw" >

    <xsd:simpleType name="defaultType">
      <xsd:restriction base="xsd:string">
      </xsd:restriction>
    </xsd:simpleType>

    <xsd:complexType name="caseType">
      <xsd:simpleContent>
        <xsd:extension base="xsd:string">
          <xsd:attribute name="value" type="xsd:string" use="required" />
        </xsd:extension>
      </xsd:simpleContent>
    </xsd:complexType >

    <xsd:complexType name="switchType" >
      <xsd:sequence>
        <xsd:element name="case" minOccurs="0" maxOccurs="unbounded" type="hw:caseType" />
        <xsd:sequence minOccurs="0" maxOccurs="1">
          <xsd:element name="default" type="hw:defaultType" />
          <xsd:element name="case" minOccurs="0" maxOccurs="unbounded" type="hw:caseType" />
        </xsd:sequence>
      </xsd:sequence>
      <xsd:attribute name="var" type="xsd:string" use="required" />
    </xsd:complexType>

  <xsd:complexType name="configType" >
    <xsd:sequence>
      <xsd:element name="switch" type="hw:switchType" />
    </xsd:sequence>
    <xsd:attribute name="name" type="xsd:string" />
  </xsd:complexType>

  <xsd:element name="config" type="hw:configType" />

</xsd:schema>

为什么这个架构不是决定论者?是否有任何方法可以将其改为确定性?

1 个答案:

答案 0 :(得分:0)

正如其他人所指出的,这似乎是xmllint(或更确切地说是libxml)中的错误。等效的DTD没有任何问题 - 即声明

<!ELEMENT switch (case*, (default, case*)?) >

被接受,没有投诉。所以看起来这个bug纯粹是在XSD的支持下,而不是更普遍的确定性检查。

一种解决方法是更换(...)?可以选择(默认值,大小写*)和空序列:

<xsd:complexType name="switchType" >
  <xsd:sequence>
    <xsd:element name="case" 
                 minOccurs="0" maxOccurs="unbounded" 
                 type="hw:caseType" />
    <xsd:choice>
      <xsd:sequence/>
      <xsd:sequence>
        <xsd:element name="default" type="hw:defaultType" />
        <xsd:element name="case" 
                     minOccurs="0" maxOccurs="unbounded" 
                     type="hw:caseType" />
      </xsd:sequence>
    </xsd:choice>
  </xsd:sequence>
  <xsd:attribute name="var" type="xsd:string" use="required" />
</xsd:complexType>