我的XML看起来像:
<?xml version=\"1.0\"?>
<itemSet>
<Item>one</Item>
<Item>two</Item>
<Item>three</Item>
.....maybe more Items here.
</itemSet>
Run Code Online (Sandbox Code Playgroud)
某些个别物品可能存在也可能不存在.假设我想要检索元素<Item>2,</Item>如果它存在.我尝试了以下XPath(在C#中).
XMLNode node = myXMLdoc.SelectSingleNode("/itemSet[Item='two']")---如果存在第2项,那么它只返回第一个第一个元素.也许这个查询只指向itemSet中的第一个元素,如果它有一个值为2的Item作为子元素.这种解释是否正确?所以我尝试过:
XMLNode node = myXMLdoc.SelectSingleNode("/itemSet[Item='two']/Item[1]")---我读了这个查询,返回<Item>itemSet中第一个有value ='two'的元素.我对么?这仍然只返回的第一个元素一个.我究竟做错了什么?在这两种情况下,使用兄弟姐妹我可以遍历子节点并获得两个,但这不是我正在看的.如果两个不存在,则SelectSingleNode返回null.因此,我获得一个成功的返回节点的事实确实表明存在元素二,所以如果我想要一个布尔测试来存在两个,上面的任何XPath就足够了,但实际上我需要完整的元素<Item>two</Item>作为我的返回节点.
[我的第一个问题,也是我第一次使用网络编程,所以我刚刚从过去的SO问题中学到了上面的XPath和相关的xml内容.所以请保持温和,让我知道如果我是一个doofus或蔑视任何社区规则.谢谢.]
Jon*_*eet 23
我想你想要:
myXMLdoc.SelectSingleNode("/itemSet/Item[text()='two']")
Run Code Online (Sandbox Code Playgroud)
换句话说,您需要具有两个文本的Item,而不是itemSet包含它的Item.
在您的情况下,您还可以使用单个点来指示上下文节点:
myXMLdoc.SelectSingleNode("/itemSet/Item[.='two']")
Run Code Online (Sandbox Code Playgroud)
编辑:.和之间的区别text()是.有效地表示"此节点",并且text()意味着"此节点的所有文本节点子节点".在这两种情况下,比较都将与LHS的"字符串值"相对照.对于元素节点,string-value是"文档顺序中元素节点的所有文本节点后代的字符串值的串联",对于文本节点的集合,比较将检查是否有任何文本节点等于你要测试的那个.
因此,元素内容何时只有一个文本节点并不重要,但假设我们有:
<root>
<item name="first">x<foo/>y</item>
<item name="second">xy<foo/>ab</item>
</root>
Run Code Online (Sandbox Code Playgroud)
然后XPath表达式" root/item[.='xy']"将匹配第一个项目,但" root/item[text()='xy']"将匹配第二个项目.