C# 对错误信息的混淆;选择器或字段的XPath无效;

C# 对错误信息的混淆;选择器或字段的XPath无效;,c#,xml,xpath,xsd,C#,Xml,Xpath,Xsd,我试图更好地理解key和keyref以及如何在模式中使用它们。我希望将密钥应用于y类型的零件,但不应用于z类型的零件。我不明白为什么我会看到以下错误 “r:B/r:part[@type='y']”是选择器或字段的无效XPath 我几乎可以肯定XPath是有效的,但在XPath表达式中输入谓词过滤器后,VisualStudio就会发出一个错误 这是XML <root xmlns="namespace1"> <A> <!-- if the ref-numbe

我试图更好地理解key和keyref以及如何在模式中使用它们。我希望将密钥应用于y类型的零件,但不应用于z类型的零件。我不明白为什么我会看到以下错误

“r:B/r:part[@type='y']”是选择器或字段的无效XPath

我几乎可以肯定XPath是有效的,但在XPath表达式中输入谓词过滤器后,VisualStudio就会发出一个错误

这是XML

<root xmlns="namespace1">
  <A>
    <!-- if the ref-number is not equal to one of the key-number, the validation will give error -->
    <part ref-number="1"/>
  </A>
  <A>
    <!-- if the ref-number is not equal to one of the key-number, the validation will give error -->
    <part ref-number="2"/>
  </A>
  <B>
    <part type="y" key-number="1"/>
    <part type="y" key-number="2"/>
    <part type="z" key-number="3"/>
  </B>
</root>

这是模式

    <?xml version="1.0" encoding="utf-8"?>
    <xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema" targetNamespace="namespace1" xmlns:r="namespace1" elementFormDefault="qualified">
      <xs:element name="root">
        <xs:complexType>
          <xs:sequence>
            <xs:element name="A" type="r:aType" maxOccurs="unbounded">
              <xs:keyref name="dummy" refer="r:partNumberKey">
                <xs:selector xpath="r:part"/>
                <xs:field xpath="@ref-number"/>
              </xs:keyref>
            </xs:element>
            <xs:element name="B" type="r:bType"/>
          </xs:sequence>
        </xs:complexType>
        <xs:key name="partNumberKey">
<!-- I get an error here -->
          <xs:selector xpath="r:B/r:part[@type='y']"/>
          <xs:field xpath="@key-number"/>
        </xs:key>
      </xs:element>
      <xs:complexType name="aType">
        <xs:sequence>
          <xs:element name="part" maxOccurs="unbounded">
            <xs:complexType>
              <xs:simpleContent>
                <xs:extension base="xs:string">
                  <xs:attribute name="ref-number" type="xs:integer"/>
                </xs:extension>
              </xs:simpleContent>
            </xs:complexType>
          </xs:element>
        </xs:sequence>
      </xs:complexType>
      <xs:complexType name="bType">
        <xs:sequence>
          <xs:element name="part" maxOccurs="unbounded">
            <xs:complexType>
              <xs:simpleContent>
                <xs:extension base="xs:string">
                  <xs:attribute name="key-number" type="xs:integer"/>
                  <xs:attribute name="type" type="xs:string"/>
                </xs:extension>
              </xs:simpleContent>
            </xs:complexType>
          </xs:element>
        </xs:sequence>
      </xs:complexType>
    </xs:schema>

您会从Xerces/Oxygen收到一条更有意义的错误消息:

[Xerces]c-general-xpath:表达式“r:B/r:part[@type='y']”对于XML架构支持的xpath子集无效

XML模式只支持一个。以下规则列出了允许作为
xs:selector
xpath
属性值的内容:

[1]     Selector       ::=      Path ( '|' Path )*
[2]     Path       ::=      ('.//')? Step ( '/' Step )*
[3]     Step       ::=      '.' | NameTest
[4]     NameTest       ::=      QName | '*' | NCName ':' '*'
这是一种隐晦的说法,表示在此上下文中不支持带有谓词(介于
[
]
之间)的表达式

如果您自己设计了此XML,则可以为
部分
元素指定不同的名称,而不是不同的
类型
属性


另一种方法是在执行此检查的XML模式中包含Schematron规则

使用嵌入式Schematron修改XML模式

<?xml version="1.0" encoding="utf-8"?>
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema"
    xmlns:r="namespace1"
    xmlns:sch="http://purl.oclc.org/dsdl/schematron"
    targetNamespace="namespace1"
    elementFormDefault="qualified">

    <xs:annotation>
        <xs:appinfo>
            <sch:ns uri="namespace1"
                prefix="r" />
            <sch:pattern>
                <sch:rule context="r:A/r:part">
                    <sch:assert test="@ref-number = /r:root/r:B/r:part/@key-number">Key Error!</sch:assert>
                </sch:rule>
            </sch:pattern>
        </xs:appinfo>
    </xs:annotation>

    <xs:element name="root">
        <xs:complexType>
            <xs:sequence>
                <xs:element name="A" type="r:aType" maxOccurs="unbounded"/>
                <xs:element name="B" type="r:bType"/>
            </xs:sequence>
        </xs:complexType>
    </xs:element>
    <xs:complexType name="aType">
        <xs:sequence>
            <xs:element name="part" maxOccurs="unbounded">
                <xs:complexType>
                    <xs:simpleContent>
                        <xs:extension base="xs:string">
                            <xs:attribute name="ref-number" type="xs:integer"/>
                        </xs:extension>
                    </xs:simpleContent>
                </xs:complexType>
            </xs:element>
        </xs:sequence>
    </xs:complexType>
    <xs:complexType name="bType">
        <xs:sequence>
            <xs:element name="part" maxOccurs="unbounded">
                <xs:complexType>
                    <xs:simpleContent>
                        <xs:extension base="xs:string">
                            <xs:attribute name="key-number" type="xs:integer"/>
                            <xs:attribute name="type" type="xs:string"/>
                        </xs:extension>
                    </xs:simpleContent>
                </xs:complexType>
            </xs:element>
        </xs:sequence>
    </xs:complexType>
</xs:schema>

在XPath计算器中,R:前缀是已知的吗?它可能在XML中定义,但在Xpath中却不是这样。您可以尝试“详细模式”,看看这是否是问题所在。(例如。*[local-name()=B和namespace-uri()='namespace1']/…等)没有命名空间:这将产生第二个“y”:
/root/B/part[@type='y'][2]/@type
<?xml-model href="../Desktop/key.xsd" type="application/xml" schematypens="http://purl.oclc.org/dsdl/schematron"?>