XML Schema:我是否可以使某些属性的值成为必需但仍然允许其他值?

时间:2010-05-05 23:00:49

标签: xml schema xsd

(注意:我无法更改我收到的XML的结构。我只能更改验证方式。)

假设我可以像这样得到XML:

<Address Field="Street" Value="123 Main"/>
<Address Field="StreetPartTwo" Value="Unit B"/>
<Address Field="State" Value="CO"/>
<Address Field="Zip" Value="80020"/>
<Address Field="SomeOtherCrazyValue" Value="Foo"/>

我需要创建一个XSD架构,验证“街道”,“状态”和“Zip” 必须 是否存在。但我不在乎“StreetPartTwo”和/或“SomeOtherCrazyValue”是否恰好也存在。

如果我知道 我可以包含三个(并且每个只包括一次),我可以这样做:

<xs:element name="Address" type="addressType" maxOccurs="unbounded" minOccurs="3"/>

<xs:complexType name="addressType">
  <xs:attribute name="Field" use="required">
    <xs:simpleType>
      <xs:restriction base="xs:string">
        <xs:enumeration value="Street"/>
        <xs:enumeration value="State"/>
        <xs:enumeration value="Zip"/>
      </xs:restriction>
    </xs:simpleType>
  </xs:attribute>
</xs:complexType>

但这对我的情况不起作用,因为我也可能会收到我不关心的其他地址元素(也有“Field”属性)。

任何想法如何确保我关心的东西都存在,但也让其他东西也存在?

TIA! 肖恩

2 个答案:

答案 0 :(得分:6)

您只能使用XML Schema进行所需的验证。

根据"XML Schema Part 1: Structures" specification ...

  

当直接或间接包含在模型组的{粒子}中的两个或多个粒子具有相同名称的元素声明时   作为他们的{term},这些声明的类型定义必须是   相同。

并不是说您无法构建可验证正确文档的架构。这意味着,您无法构建无法验证某些不正确文档的架构。当我说“不正确”时,我指的是违反您用英语说明的限制的文件。

例如,假设您有一个包含三个Street元素的文档,如下所示:

<Address Field="Street" Value="123 Main"/> 
<Address Field="Street" Value="456 Main"/> 
<Address Field="Street" Value="789 Main"/> 
<Address Field="SomeOtherCrazyValue" Value="Foo"/> 

根据您的架构,该文档是有效的地址。可以在架构中添加xs:unique约束,以便拒绝这些损坏的文档。但即使使用xs:unique,对这样的模式进行验证也会声明其他一些不正确的文档是有效的 - 例如,包含三个<Address>元素的文档,每个元素都有一个唯一的Field属性,但是其中没有一个Field="Zip"

事实上,不可能生成一个W3C XML Schema,它正式编写了您所声明的约束。 <xs:all>元素几乎会让您受到影响,但它仅适用于元素,而不适用于属性。并且,它不能与扩展一起使用,因此您不能在W3C XML Schema中说“所有这些元素以任何顺序加上任何其他元素”。


为了执行您寻求的验证,您的选择是:

  1. 依赖于XML Schema以外的东西,
  2. 在多个步骤中执行验证,第一步使用XML Schema,第二步使用其他内容。
  3. 对于第一个选项,我认为你可以使用Relax NG来做到这一点。其缺点是,它不是一个标准,据我所知,它既没有得到广泛支持也没有增长。这就像学习盖尔语来表达思想一样。盖尔语没有错,但它是一种语言的死胡同,I think RelaxNG is, too

    对于第二个选项,一种方法是作为第一步验证您的架构,然后,作为第二步:

    一个。应用XSL转换,将<Address>元素转换为为其Field属性的值命名的元素。该转换的输出如下所示:

    <root>
      <Street Value="101 Bellavista Drive"/>
      <State  Value="Confusion"/>
      <Zip    Value="10101"/>
    </root>
    

    B中。根据不同的模式验证该变换的输出,如下所示:

    <xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema"
               elementFormDefault="qualified">
      <xs:element name="root">
        <xs:complexType>
          <xs:all>
            <xs:element maxOccurs="1" minOccurs="1" ref="Street" />
            <xs:element maxOccurs="1" minOccurs="1" ref="State" />
            <xs:element maxOccurs="1" minOccurs="1" ref="Zip" />
          </xs:all>
        </xs:complexType>
      </xs:element>
    
      <xs:element name="Street">
        <xs:complexType>
          <xs:attribute name="Value" use="required" type="xs:string"/>
        </xs:complexType>
      </xs:element>
      <xs:element name="State">
        <xs:complexType>
          <xs:attribute name="Value" use="required" type="xs:string"/>
        </xs:complexType>
      </xs:element>
      <xs:element name="Zip">
        <xs:complexType>
          <xs:attribute name="Value" use="required" type="xs:string"/>
        </xs:complexType>
      </xs:element>
    
    </xs:schema>
    

    您需要扩展该架构以处理转换输出中的其他元素,如<SomeOtherCrazyValue>。或者你可以将xsl变换结构化为不发出不属于{State,Street,Zip}的元素。

    为了清楚起见,我知道您无法更改收到的XML。这种方法不需要这样做。它只是使用时髦的两步验证方法。第二个验证步骤完成后,您可以放弃转换的结果。


    编辑 - 实际上,Sean,再次考虑这个问题,您可以使用步骤B.假设您的XSL转换只是从文档中移除 <Address> Field属性值没有State,Street或Zip的元素。换句话说,没有<Address Field="SomeOtherCrazyValue"...>。可以使用您的模式验证该变换的结果,使用maxOccurs =“3”,minOccurs =“3”和xs:unique。

答案 1 :(得分:0)

我遇到了和你一样的问题,但却用技巧克服了。

请求XML

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8"
indent="yes" />
<xsl:template match="@* | node()">
<xsl:copy>
<xsl:apply-templates select="@* | node()" />
</xsl:copy>
</xsl:template>
<xsl:template match="arg">
<xsl:element name="{@name}">
<xsl:apply-templates />
</xsl:element>
</xsl:template>
</xsl:stylesheet>

现在使用xslt并将其转换,

XSLT代码

<?xml version="1.0" encoding="UTF-8"?>
<request>

    <url>abcd</url>
    <base-url>XXXXXXX</base-url>
    <args src="url">
        <languageCode>NL</languageCode>
        <version>1</version>
        <offerId>10</offerId>
        <rewardId>1234</rewardId>
    </args>
</request>

XML变为

<xs:schema attributeFormDefault="unqualified" elementFormDefault="qualified" xmlns:xs="http://www.w3.org/2001/XMLSchema">
  <xs:element name="request">
    <xs:complexType>
      <xs:sequence>
        <xs:element type="xs:string" name="url"/>
        <xs:element type="xs:string" name="base-url"/>
        <xs:element name="args" maxOccurs="unbounded" minOccurs="0">
          <xs:complexType mixed="true">
            <xs:sequence>
              <xs:element type="xs:string" name="languageCode" minOccurs="0"/>
              <xs:element type="xs:string" name="version" minOccurs="0"/>
              <xs:element type="xs:string" name="offerId" minOccurs="1"/>
              <xs:element type="xs:string" name="rewardId" minOccurs="1"/>
            </xs:sequence>
            <xs:attribute type="xs:string" name="src" use="optional"/>
          </xs:complexType>
        </xs:element>
      </xs:sequence>
    </xs:complexType>
  </xs:element>
</xs:schema>

现在应用XSD

URI address = new URI("http://192.168.30.80/" + str);
HttpClient client = new DefaultHttpClient();
HttpGet request = new HttpGet();
request.setURI(address);
HttpResponse response = client.execute(request);