使用XPath查询带有命名空间(有xmlns)的XML(转)

使用XPath查询带有命名空间(有xmlns)的XML

标签: xmlsilverlightwebserviceencodingwpfinclude

2012-06-19 10:26 3235人阅读 评论(0) 收藏 举报

 分类:

XML(7) 

最近碰到一个小问题,通过调用webservice返回如下的xml,

<Seller xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
  <id xmlns="http://www.aaa.com/">348388</id>
  <gender xmlns="http://www.aaa.com/">Monsieur</gender>
  <lastName xmlns="http://www.aaa.com/">FABRIZZI</lastName>
  <firstName xmlns="http://www.aaa.com/">FRANCIS</firstName>
  <login xmlns="http://www.aaa.com/">[email protected]</login>
  <password xmlns="http://www.aaa.com/">KUVOTFP</password>
  <profile xmlns="http://www.aaa.com/">Chef de vente</profile>
  <email xmlns="http://www.aaa.com/">[email protected]</email>
  <mobilePhone xmlns="http://www.aaa.com/">0600000000</mobilePhone>
  <dateRec xmlns="http://www.aaa.com/">2012-02-21T22:49:49.283</dateRec>
  <dateMod xmlns="http://www.aaa.com/">2012-02-21T22:49:49.283</dateMod>
  <idDrv xmlns="http://www.aaa.com/">233734</idDrv>
  <drv xmlns="http://www.aaa.com/">DAVID ROUSSEAU</drv>
  <idCrv xmlns="http://www.aaa.com/">233775</idCrv>
  <crv xmlns="http://www.aaa.com/">ALEXANDRA CORDIER</crv>
  <idCdv xmlns="http://www.aaa.com/">0</idCdv>
  <dealerCode xmlns="http://www.aaa.com/">FB001</dealerCode>
  <dealerType xmlns="http://www.aaa.com/">Premium</dealerType>
  <dealerQualification xmlns="http://www.aaa.com/">GOLD</dealerQualification>
  <dealerName xmlns="http://www.aaa.com/">BAYERN AIX</dealerName>
  <dealerAddress xmlns="http://www.aaa.com/">ZA La Pioline</dealerAddress>
  <dealerZipCode xmlns="http://www.aaa.com/">13546</dealerZipCode>
  <dealerCity xmlns="http://www.aaa.com/">AIX EN PROVENCE</dealerCity>
  <dealerPhone1 xmlns="http://www.aaa.com/">0442162070</dealerPhone1>
  <dealerPhone2 xmlns="http://www.aaa.com/">0442162089</dealerPhone2>
  <dealerSellChannel xmlns="http://www.aaa.com/">CCP BMW Mini</dealerSellChannel>
</Seller>

当时也没有多想,直接用xpath解析了,没想到死活不成功,再仔细一看,原来有命名空间,真是头大。

然后去google找答案,如下部分转自帖子http://www.cnblogs.com/mgen/archive/2011/05/24/2056025.html

众所周知,XmlDocument可以进行XPath查询,但实际上这里所说的XPath查询仅限于没有命名空间(没有xmlns属性)的XML,一旦遇到有命名空间的XML,对应XPath查询都会无结果。

比如下面这个XML

<a xmlns="mgen.cnblogs.com">

<b>ccc</b>

</a>

XPath查询/a/b会返回null,而如果没有xmlns的话,会返回节点b。

为什么会这样呢?MSDN的相应函数有解释(参考:http://msdn.microsoft.com/en-us/library/system.xml.xmlnode.selectsinglenode.aspx

If the XPath expression does not include a prefix, it is assumed that the namespace URI is the empty namespace. If your XML includes a default namespace, you must still add a prefix and namespace URI to the XmlNamespaceManager; otherwise, you will not get any nodes selected

意思就是如果XPath表达式没有加前缀(如a:b中前缀是a),那么所查询节点(注意属性也可以是节点)的命名空间URI就应该是空值(也是默认值),否则XPath不会返回结果。

上面的XML, 因为节点a和b都有命名空间值,自然XPath查询不会有结果。

(上面英文还提到如果节点有默认命名空间,那么还得手动向XmlNamespaceManager添加前缀和命名空间值,这个在后面会讲的)

在看解决方案前,首先需要能够辨识XML命名空间,当然辨识XML命名空间值还是很容易的,参考如下XML(这个XML在后面程序中也会用到)

<?xml version="1.0" encoding="utf-8"?>

<root xmlns="dotnet" xmlns:w="wpf">

<a>data in a</a>

<w:b>data in b</w:b>

<c xmlns="silverlight">

<w:d>

<e>data in e</e>

</w:d>

</c>

</root>

它的所有XML节点的命名空间如下所示:

<?xml version="1.0" encoding="utf-8"?>

<root xmlns="dotnet" xmlns:w="wpf">

<!-- xmlns: dotnet -->

<a>data in a</a>

<!-- xmlns: dotnet -->

<w:b>data in b</w:b>

<!-- xmlns: wpf -->

<c xmlns="silverlight">

<!-- xmlns: silverlight -->

<w:d>

<!-- xmlns: wpf -->

<e>data in e</e>

<!-- xmlns: silverlight -->

</w:d>

</c>

</root>

如果识别XML命名空间没有问题,那么后面的操作就相当简单了,你需要记住:在XmlDocument中用XPath查询某一节点时,只要它的命名空间值不是空值,那么你必须给它一个前缀,用这个前缀代表这个节点的命名空间值!这些前缀是通过XmlNamespaceManager类添加的,使用时将XmlNamespaceManager传入SelectNodes或SelectSingleNode中即可。这也是为什么上面说“如果节点有默认命名空间,那么还得手动向XmlNamespaceManager添加前缀和命名空间值”的原因。

另外构造一个XmlNamespaceManager需要XmlNameTable对象,这个对象可以从XmlDocument.NameTable和XmlReader.NameTable属性中得到。

下面我们步入代码,比如说查询上面XML中的节点e,分析位置节点e位于:root->c->d->e,然后将所需命名空间值加入到XmlNamespaceManager中(前缀名称无所谓,只要在XPath一致即可),查询即可成功,如下代码:

/*

* 假设上面XML文件在C:\a.txt中

* 下面代码会查询目标节点e,并输出数据:data in e

* */

var xmlDoc = new XmlDocument();

xmlDoc.Load(@"C:\a.txt");

//加入命名空间和前缀

var xmlnsm = new XmlNamespaceManager(xmlDoc.NameTable);

xmlnsm.AddNamespace("d", "dotnet");

xmlnsm.AddNamespace("s", "silverlight");

xmlnsm.AddNamespace("w", "wpf");

var node = xmlDoc.SelectSingleNode("/d:root/s:c/w:d/s:e", xmlnsm);

Console.WriteLine(node.InnerText);

//输出:data in e

时间: 2024-08-05 07:52:04

使用XPath查询带有命名空间(有xmlns)的XML(转)的相关文章

利用XPath解析带有xmlns的XML文件

在.net中,编写读取xml 的程序中提示"未将对象引用设置到对象的实例",当时一看觉得有点奇怪.为什么在读取xml数据的时候也要实例化一个对象.google了才知道,xml文件中加入了xmlns表示名称空间,但同时Xpath也必须加上. 如之前我们的xml文件定义为: <Project Name="目标计划项目"> <Process Name="个人月度计划" Description="个人月度计划" Ve

c#读取带有命名空间的XML

XML文件内容 <?xml version="1.0" standalone="no"?> <svg width="1709px" height="886px" onload="init(evt)" viewBox="0 0 1800 900" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="

带有命名空间的xml解析,C#

前一段时间做花旗的接口,返回的xml格式是带有命名空间的,可是难倒了我,找了好久才找到解决办法,给大家分享下,少走弯路. 1,直接进入正题,先看一段带有命名空间的xml,这段xml大概的意思是,前面是header信息,后面是body中的信息,body中有不少的节点,主要的是 <ns6:RewardDetail>节点,这个节点包含了很多的<ns8:RedemptionDetails>节点,和一个<ns8:CurrentRewardStatementDetails>节点以及

php解析带有命名空间的xml

xml如果带有命名空间我们将如何解析,例如: <ns1:CreateBillResponse xmlns:ns1="http://neusoft.com" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/" xmlns:xsi="http://www.w3.org/2001/XMLSc

Java使用XPath查询XML中的元素

使用java中的xpath语法查询xml中元素 1.查询的xml文件为student.xml文件 <?xml version="1.0" encoding="UTF-8" ?><students> <student class="1201">  <name>李小离</name>  <sex>男</sex>  <age>25</age> &l

python 使用xpath解析含有命名空间(xmlns)的xml

解决办法: from lxml import etree xml = etree.parse("./cee.xml") root = xml.getroot() print(root.xpath(".//i:Reviews", namespaces={"i":"http://www.bazaarvoice.com/xs/PRR/StandardClientFeed/14.7"})) 更多语言参考 https://stackov

xml(带有命名空间的)读写操作

xml文件: <?xml version="1.0" encoding="UTF-8"?><!-- This file contains job definitions in schema version 2.0 format --><job-scheduling-data xmlns="http://quartznet.sourceforge.net/JobSchedulingData" xmlns:xsi=&qu

Python通过lxml库遍历xml通过xpath查询(标签,属性名称,属性值,标签对属性)

xml实例: 版本一: <?xml version="1.0" encoding="UTF-8"?><country name="chain"><provinces><heilongjiang name="citys"><haerbin/><daqing/></heilongjiang><guangdong name="city

命名空间(xmlns属性)

xmlns 属性可以在文档中定义一个或多个可供选择的命名空间.该属性可以放置在文档内任何元素的开始标签中.该属性的值类似于 URL,它定义了一个命名空间,浏览器会将此命名空间用于该属性所在元素内的所有内容. 例如,如果需要使用符合 XML 规范的 XHTML 文档,则应该在文档中的<html> 标签中至少使用一个 xmlns 属性,以指定整个文档所使用的主要命名空间: <html xmlns="http://www.w3.org/1999/xhtml"> 如果需