13

我的 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 的项作为子项。这种解释正确吗?

所以我尝试了:

  • XMLNode node = myXMLdoc.SelectSingleNode("/itemSet[Item='two']/Item[1]")--- 我将这个查询读作,返回<Item>itemSet 中第一个值 = 'two' 的元素。我对么?

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

[我在这里的第一个问题,也是我第一次使用 Web 编程,所以我刚刚从过去的 SO 问题中学习了上述 XPath 和相关的 xml 内容。所以要温柔,让我知道我是个傻瓜还是无视任何社区规则。谢谢。]

4

1 回答 1

23

我想你想要:

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

换句话说,您想要具有两个文本的项目itemSet,而不是包含它的项目。

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

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

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

所以当元素内容只有一个文本节点时并不重要,但假设我们有:

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

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

于 2009-05-19T16:17:25.140 回答