我正在尝试从流阅读器中读取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>
答案 0 :(得分:36)
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="..."
。这意味着除非特别重写,否则所有元素也会使用此命名空间。
答案 1 :(得分:9)
您可以在读取文件时删除命名空间,只需在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时,不必关心命名空间。
答案 2 :(得分:8)
问题在于您要求的RateQuote
元素没有名称空间 - 而RateQuote
元素在名称空间中 http://ratequote.usfnet.usfc.com/v2/x1
您可以 使用XmlNamespaceManager
来解决XPath中的命名空间,或者使用LINQ to 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");
答案 3 :(得分:5)
您可能希望在XmlTextReader中将命名空间设置为 false 。
因此,在您的代码中,更改:
var xmlReader = new XmlTextReader(stream);
到
var xmlReader = new XmlTextReader(stream) { Namespaces = false };
通过此更改,您应该能够使用SelectSingleNode获取所需的节点,而无需使用名称空间。
答案 4 :(得分:1)
您还应该能够:
...
var node = xmlDocument["RateQuote"];
...
VB的语法是:
...
Dim node as XmlNode = xmlDocument("RateQuote")
...