1

我正在尝试使用 .NET 解析 c# .NET 中的有效 XML 文档xPathNavigator。XML 文档的开头如下所示:

<soapenv:Envelope xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
    <soapenv:Body>
        <ns1:myResponse xmlns:ns1="ExampleNS" soapenv:encodingStyle="http://schemas.xmlsoap.org/soap/encoding/">
            <myReturn href="#2" />

如您所见,在根元素上定义了一些命名空间,但稍后ns1会定义。当我尝试评估 xPath 查询时/soapenv:Envelope/soapenv:Body/ns1:myResponse/myReturn,我得到XPathException

Namespace prefix 'ns1' is not defined.

我正在做的是将 XML 文档作为字符串加载到XmlDocument对象中。然后我正在创建一个导航器,移动到根元素并调用GetNamespacesInScope(XmlNamespaceScope.All). 我遍历这个集合,其中包含根上定义的、、和的命名空间,并将它们添加到命名空间xml管理器soapenv。然后我创建一个并调用,并得到异常。xsdxsixPathNavigatorEvaluate

代码是这样的:

string response = GetXMLString();
var xmlDocument = new XmlDocument();
xmlDocument.LoadXml(response);

var xmlDocumentNavigator = xmlDocument.CreateNavigator();
xmlDocumentNavigator.MoveToFollowing(XPathNodeType.Element);
var xnm = new XmlNamespaceManager(xmlDocument.NameTable);
var namespacesInScope = xmlDocumentNavigator.GetNamespacesInScope(XmlNamespaceScope.All);
if (namespacesInScope != null)
{
    foreach (var prefix in namespacesInScope.Keys)
    {
        xnm.AddNamespace(prefix, namespacesInScope[prefix]);
    }
}

var xPathDocument = new XPathDocument(new StringReader(response));
var xPathNavigator = xPathDocument.CreateNavigator();

xPathNavigator.Evaluate("/soapenv:Envelope/soapenv:Body/ns1:myResponse/myReturn", xnm);

为什么GetNamespacesInScope方法不接ns1

4

2 回答 2

3

问题是GetNamespacesInScope只返回导航器当前节点范围内的命名空间(我猜该方法建议:)。

获取所有命名空间的一种技巧是遍历所有元素,获取命名空间,然后仅重新组装唯一的元素。

字典合并代码由 JonSkeet 提供

 XPathDocument xpd = new XPathDocument(new StringReader(s));
 XPathNavigator xpn = xpd.CreateNavigator();
 List<IDictionary<string, string>> xmlnsList = new List<IDictionary<string,string>>();
 while (xpn.MoveToFollowing(XPathNodeType.Element))
 {
     xmlnsList.Add(xpn.GetNamespacesInScope(XmlNamespaceScope.All));
 }
 var result = xmlnsList.SelectMany(dict => dict)
              .ToLookup(pair => pair.Key, pair => pair.Value)
              .ToDictionary(group => group.Key, group => group.First());
 if (result != null)
 {
     foreach (var prefix in result.Keys)
     {
         xnm.AddNamespace(prefix, result[prefix]);
     }
 }

编辑

根据 Paciv 的评论,如果您真的事先不知道 xml 文档中的命名空间是什么,另一种方法是使用与命名空间无关的 Xpath,并避免嗅探命名空间的问题(并完全消除对命名空间管理器的需要) .

在您的示例中,这将是:

xPathNavigator.Evaluate("/*[local-name()='Envelope']/*[local-name()='Body']/*[local-name()='myResponse']/*[local-name()='myReturn']");
于 2012-09-12T15:13:50.243 回答
2

只需添加您事先知道执行 XPath 查询所需的名称空间怎么样?

你可以写

xnm.AddNamespace("MyNS", "ExampleNS");

进而

xPathNavigator
    .Evaluate("/soapenv:Envelope/soapenv:Body/MyNS:myResponse/myReturn", xnm);

只要前缀设计的命名空间相同,前缀是什么都无所谓。

顺便说一句,您应该对 SOAP 名称空间执行相同的操作,因为您将 XPath 查询基于前缀,并且可以生成完全相同的 XML,具有针对相同名称空间的不同前缀。

我会写一些更健壮的东西,适用于不同的 SOAP 生成器,比如

string response = GetXMLString();
XmlDocument xmlDocument = new XmlDocument();
xmlDocument.LoadXml(response);
XmlNamespaceManager xnm = new XmlNamespaceManager(xmlDocument.NameTable);
xnm.AddNamespace("soapenv", "http://schemas.xmlsoap.org/soap/envelope/");
xnm.AddNamespace("ns1", "ExampleNS");

xmlDocument.CreateNavigator()
    .Evaluate("/soapenv:Envelope/soapenv:Body/ns1:myResponse/myReturn", xnm);
于 2012-09-12T16:10:59.657 回答