gpt4 book ai didi

c# - 为什么 NamespaceManager 在 XPath 中不使用前缀时不使用 DefaultNamespace

转载 作者:数据小太阳 更新时间:2023-10-29 02:01:31 26 4
gpt4 key购买 nike

当我想使用 XPath 遍历我的 XmlDocument 时,我遇到了文档中有许多难看的命名空间的问题,所以我开始使用 NamespaceManager 和 XPath。

XML 看起来像这样

<Workbook xmlns="urn:schemas-microsoft-com:office:spreadsheet"
xmlns:o="urn:schemas-microsoft-com:office:office"
xmlns:x="urn:schemas-microsoft-com:office:excel"
xmlns:ss="urn:schemas-microsoft-com:office:spreadsheet"
xmlns:html="http://www.w3.org/TR/REC-html40">
<Worksheet ss:Name="KA0100401">
<Table>
<Row>
<Cell>Data</Cell>
</Row>
<!-- more rows... -->
</Table>
</Worksheet>
<Worksheet ss:Name="KA0100402">
<!-- .... --->
</Worksheet>
</Workbook>

现在,根据我从该文档中看到的内容,"urn:schemas-microsoft-com:office:spreadsheet" 是默认命名空间,因为它位于根元素上。

所以,天真地,我像这样配置了我的 NamespaceManager:

XmlDocument document = new XmlDocument();
document.Load(reader);
XmlNamespaceManager manager = new XmlNamespaceManager(document.NameTable);
manager.AddNamespace(String.Empty, "urn:schemas-microsoft-com:office:spreadsheet");
manager.AddNamespace("o", "urn:schemas-microsoft-com:office:office");
manager.AddNamespace("x", "urn:schemas-microsoft-com:office:excel");
manager.AddNamespace("ss", "urn:schemas-microsoft-com:office:spreadsheet");
manager.AddNamespace("html", "http://www.w3.org/TR/REC-html40");

但是,当我尝试访问一个节点时

foreach (XmlNode row in document.SelectNodes("/Workbook/Worksheet[1]/Table/Row", manager))

我从来没有得到任何结果。我的印象是,通过将第一个命名空间设置为空前缀,在该工作区中搜索节点时就不需要设置它了。

但是,事实上stated on the AddNamespace method :

If an XPath expression does not include a prefix, it is assumed that the namespace Uniform Resource Identifier (URI) is the empty namespace.

这是为什么呢?而且,更重要的是:如果不使用前缀将它们设置为空命名空间,我如何访问默认命名空间中的节点?

如果我在搜索节点时甚至无法访问它,那么在管理器上设置默认命名空间有什么好处?

最佳答案

来自XPath 1.0 spec :

A QName in the node test is expanded into an expanded-name using the namespace declarations from the expression context. This is the same way expansion is done for element type names in start and end-tags except that the default namespace declared with xmlns is not used: if the QName does not have a prefix, then the namespace URI is null (this is the same way attribute names are expanded). It is an error if the QName has a prefix for which there is no namespace declaration in the expression context.

所以这不是关于 NamespaceManager 的问题,而是 XPath 定义工作方式的问题。


您缺少的一点是,您在 NamespaceManager 中使用的前缀不必与 XML 文档中的前缀相同。您可以根据需要为 urn:schemas-microsoft-com:office:excel 使用 xcel 前缀,为 使用 sp 前缀>urn:schemas-microsoft-com:office:spreadsheet。事实上,您已经在命名空间管理器中为该 URN 分配了一个前缀,因此您可以直接使用它:

foreach (XmlNode row in 
document.SelectNodes("/ss:Workbook/ss:Worksheet[1]/ss:Table/ss:Row", manager))


关于这个问题:

What good is setting the default namespace on the manager if I can't even access it when searching for nodes?

好处是 XmlNamespaceManager 不仅仅用于评估 XPath。例如,它可用于跟踪 XML 文档中的命名空间,其中有默认命名空间的概念。

关于c# - 为什么 NamespaceManager 在 XPath 中不使用前缀时不使用 DefaultNamespace,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26630995/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com