繁体   English   中英

XML Document SelectSingleNode返回null

[英]XML Document SelectSingleNode returns null

我试图从流阅读器读取XML,并且还获得响应XML。 但是当我尝试读取它的节点时,它总是返回null。

var request = (HttpWebRequest) WebRequest.Create(address);
var response = (HttpWebResponse) request.GetResponse();
var stream = response.GetResponseStream();

if(stream != null)
{
   var xmlReader = new XmlTextReader(stream);
   var xmlDocument = new XmlDocument();
   xmlDocument.Load(xmlReader);
   var node = xmlDocument.SelectSingleNode("RateQuote");
}

XML文档

<RateQuoteResponse xmlns="http://ratequote.usfnet.usfc.com/v2/x1">
  <STATUS>
   <CODE>0</CODE>
   <VIEW>SECURED</VIEW>
   <VERSION>...</VERSION>
  </STATUS>
 <RateQuote>
   <ORIGIN>
     <NAME>KNOXVILLE</NAME>
     <CARRIER>USF Holland, Inc</CARRIER>
     <ADDRESS>5409 N NATIONAL DR</ADDRESS>
     <CITY>KNOXVILLE</CITY>
     <STATE>TN</STATE>
     <ZIP>37914</ZIP>
     <PHONE>8664655263</PHONE>
     <PHONE_TOLLFREE>8006545963</PHONE_TOLLFREE>
     <FAX>8656379999</FAX>
  </ORIGIN>
  <DESTINATION>
     <NAME>KNOXVILLE</NAME>
     <CARRIER>USF Holland, Inc</CARRIER>
     <ADDRESS>5409 N NATIONAL DR</ADDRESS>
     <CITY>KNOXVILLE</CITY>
     <STATE>TN</STATE>
     <ZIP>37914</ZIP>
     <PHONE>8664655263</PHONE>
     <PHONE_TOLLFREE>8006545963</PHONE_TOLLFREE>
     <FAX>8656379999</FAX>
  </DESTINATION>
     <ORIGIN_ZIP>37914</ORIGIN_ZIP>
     <DESTINATION_ZIP>37909</DESTINATION_ZIP>
     <TOTAL_COST>99.24</TOTAL_COST>
     <SERVICEDAYS>1</SERVICEDAYS>
     <INDUSTRYDAYS>1.6</INDUSTRYDAYS>
     <CLASSWEIGHT>
        <CLASS>55</CLASS>
        <ASCLASS>50</ASCLASS>
        <WEIGHT>100</WEIGHT>
        <CHARGES>0.0</CHARGES>
     </CLASSWEIGHT>
</RateQuote>
</RateQuoteResponse>

XML文档使用默认命名空间“ http://ratequote.usfnet.usfc.com/v2/x1 ”。 您需要更改SelectSingleNode调用以使用此命名空间。

您需要设置一个namspace管理器,然后将其提供给SelectSingleNode

var nsmgr = new XmlNamespaceManager(doc.NameTable);
nsmgr.AddNamespace("rate", "http://ratequote.usfnet.usfc.com/v2/x1");
var node = xmlDocument.SelectSingleNode("//rate:RateQuote", nsmgr);

编辑 RateQuoteResponse元素具有默认名称空间xmlns="..." 这意味着除非特别重写,否则所有元素也会使用此命名空间。

您可以在读取文件时删除命名空间,只需禁用XmlTextReader上的命名空间:

var request = (HttpWebRequest) WebRequest.Create(address);
var response = (HttpWebResponse) request.GetResponse();
var stream = response.GetResponseStream();

if(stream != null)
{
   var xmlReader = new XmlTextReader(stream);
   xmlReader.Namespaces = false;
   var xmlDocument = new XmlDocument();
   xmlDocument.Load(xmlReader);
   var node = xmlDocument.SelectSingleNode("RateQuote");
}

之后,在XML元素上使用XPath / LINQ时,您不必关心命名空间。

问题是,你问一个RateQuote没有命名空间元素-而RateQuote元素实际上是与URI命名空间http://ratequote.usfnet.usfc.com/v2/x1

可以使用XmlNamespaceManager以满足您的XPath内的命名空间,或使用LINQ到XML具有非常简单的命名空间的处理:

var document = XDocument.Load(stream);
XNamespace ns = "http://ratequote.usfnet.usfc.com/v2/x1";
XElement rateQuote = document.Root.Element(ns + "RateQuote");

我个人会尽可能使用LINQ to XML - 我发现它比XmlDocument更令人愉快。 如果你想要的话,你仍然可以使用XPath,但我个人更喜欢使用查询方法。

编辑:请注意,名称空间默认也适用于子元素。 所以要找到你需要的TOTAL_COST元素:

XElement cost = document.Root
                        .Element(ns + "RateQuote")
                        .Element(ns + "TOTAL_COST");

您可能希望在XmlTextReader中将Namespaces设置为false

所以,在你的代码中,改变:

var xmlReader = new XmlTextReader(stream);

var xmlReader = new XmlTextReader(stream) { Namespaces = false };

通过此更改,您应该能够使用SelectSingleNode获取所需的节点,而无需使用命名空间。

你还应该能够做到:

...
var node = xmlDocument["RateQuote"];
...

VB的语法是:

...
Dim node as XmlNode = xmlDocument("RateQuote")
...

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM