在SelectSingleNode中使用XPath:从XML中检索单个元素(如果存在)

时间:2009-05-19 16:12:07

标签: c# xpath selectsinglenode

我的XML看起来像:

<?xml version=\"1.0\"?>
<itemSet>
       <Item>one</Item>
       <Item>two</Item>
       <Item>three</Item>
       .....maybe more Items here.
</itemSet>

某些个人项目可能存在也可能不存在。假设我想检索元素 <Item> 两个 </Item> (如果存在)。我尝试了以下XPath(在C#中)。

  • XMLNode node = myXMLdoc.SelectSingleNode("/itemSet[Item='two']") ---如果Item two 存在,那么它只返回第一个元素 one 。也许这个查询只指向itemSet中的第一个元素,如果它在某个地方有一个值为2的Item作为子元素。这种解释是否正确?

所以我试过了:

  • XMLNode node = myXMLdoc.SelectSingleNode("/itemSet[Item='two']/Item[1]") ---我将此查询视为,将itemSet中第一个 <Item> 元素返回给我,其值为''two'。我是对的吗?

这仍然只返回第一个元素 one 。我究竟做错了什么? 在这两种情况下,使用兄弟姐妹我可以遍历子节点并到达两个,但这不是我正在看的。如果两个不存在,则SelectSingleNode返回null。因此,我获得一个成功的返回节点的事实确实表明存在元素二,所以如果我想要一个布尔测试来chk存在两个,任何上述XPath就足够了,但是我实际上需要将完整元素 <Item>two</Item> 作为我的返回节点。

[我的第一个问题,也是我第一次使用网络编程,所以我刚刚从过去的SO问题中学到了上面的XPath和相关的xml内容。所以请保持温和,让我知道如果我是一个doofus或蔑视任何社区规则。感谢。]

1 个答案:

答案 0 :(得分:23)

我想你想要:

myXMLdoc.SelectSingleNode("/itemSet/Item[text()='two']")

换句话说,您希望 Item 的文本为2,而不是包含它的itemSet

在您的情况下,您还可以使用单个点来指示上下文节点:

myXMLdoc.SelectSingleNode("/itemSet/Item[.='two']")

编辑:.text()之间的区别在于.有效地表示“此节点”,text()表示“此节点的所有文本节点子节点” 。在这两种情况下,比较都将与LHS的“字符串值”相对照。对于元素节点,string-value是“文档顺序中元素节点的所有文本节点后代的字符串值的串联”,对于文本节点的集合,比较将检查是否有任何文本节点等于你要测试的那个。

因此,元素内容只有一个文本节点并不重要,但假设我们有:

<root>
  <item name="first">x<foo/>y</item>
  <item name="second">xy<foo/>ab</item>
</root>

然后XPath表达式“root/item[.='xy']”将匹配第一项,但“root/item[text()='xy']”将匹配第二项。