从webrequest解析xmldocument

时间:2013-07-31 16:05:48

标签: c# xml linq xmldocument

我正好解析这个布局:

<string xmlns="http://www.namespaceuri.com/Admin/ws">
    <CardTrxSummary> 
        <PaymentMethod> 
            <Payment_Type_ID>VISA </Payment_Type_ID> 
            <Authorization>0.0000</Authorization> 
            <Capture>0.0000</Capture> <ForceCapture>0.0000</ForceCapture> 
            <PostAuth>0.0000</PostAuth> <Return>0.0000</Return> 
            <Sale>3419.2700</Sale> <Receipt>0.0000</Receipt> 
            <RepeatSale>0.0000</RepeatSale> 
            <Activate>0.0000</Activate> 
            <Deactivate>0.0000</Deactivate> 
            <Reload>0.0000</Reload> 
            <Authorization_Cnt>0</Authorization_Cnt> 
            <Capture_Cnt>0</Capture_Cnt> 
            <ForceCapture_Cnt>0</ForceCapture_Cnt> 
            <PostAuth_Cnt>0</PostAuth_Cnt> 
            <Return_Cnt>0</Return_Cnt> 
            <Sale_Cnt>13</Sale_Cnt> 
            <Receipt_Cnt>0</Receipt_Cnt> 
            <RepeatSale_Cnt>0</RepeatSale_Cnt> 
            <Activate_Cnt>0</Activate_Cnt> 
            <Deactivate_Cnt>0</Deactivate_Cnt> 
            <Reload_Cnt>0</Reload_Cnt> 
            <Cnt>13</Cnt> 
        </PaymentMethod> 
    </CardTrxSummary>
</string>

我正在尝试使用此代码来获取特定结果:

private static string ReadValueFromXml(XmlDocument xmlDocument, string field)
{
  var xdoc = xmlDocument.ToXDocument();
  var ns = "http://www.namespaceuri.com/Admin/ws";
  return xdoc.Descendants(ns + "PaymentMethod")
             .Select(x => (string) x.Attribute("Cnt"))
             .FirstOrDefault();
}

此时,它正在给我这条消息:

  

':'字符,十六进制值0x3A,不能包含在a中   名。

我试过这种方式:

  XmlNodeList xnList = xmlDocument.SelectNodes("/CardTrxSummary/PaymentMethod");
  foreach (XmlNode xn in xnList)
  {
    Console.WriteLine("Sale: " + xn["Sale"].InnerText);
    Console.WriteLine("Sale_Cnt: " + xn["Sale_Cnt"].InnerText);
    Console.WriteLine("Payment_Type_ID: " + xn["Payment_Type_ID"].InnerText);
  }

它永远不会进入foreach。

如何获取PaymentMethod中的值?

修改

我查看了xmldocument的innertext,这就是它的显示方式:

"<?xml version=\"1.0\" encoding=\"utf-8\"?><string xmlns=\"http://www.namespaceuri.com/Admin/ws\">&lt;CardTrxSummary&gt;\r\n  &lt;PaymentMethod&gt;\r\n    &lt;Payment_Type_ID&gt;VISA      &lt;/Payment_Type_ID&gt;\r\n    &lt;Authorization&gt;0.0000&lt;/Authorization&gt;\r\n    &lt;Capture&gt;0.0000&lt;/Capture&gt;\r\n    &lt;ForceCapture&gt;0.0000&lt;/ForceCapture&gt;\r\n    &lt;PostAuth&gt;0.0000&lt;/PostAuth&gt;\r\n    &lt;Return&gt;0.0000&lt;/Return&gt;\r\n    &lt;Sale&gt;3419.2700&lt;/Sale&gt;\r\n    &lt;Receipt&gt;0.0000&lt;/Receipt&gt;\r\n    &lt;RepeatSale&gt;0.0000&lt;/RepeatSale&gt;\r\n    &lt;Activate&gt;0.0000&lt;/Activate&gt;\r\n    &lt;Deactivate&gt;0.0000&lt;/Deactivate&gt;\r\n    &lt;Reload&gt;0.0000&lt;/Reload&gt;\r\n    &lt;Authorization_Cnt&gt;0&lt;/Authorization_Cnt&gt;\r\n    &lt;Capture_Cnt&gt;0&lt;/Capture_Cnt&gt;\r\n    &lt;ForceCapture_Cnt&gt;0&lt;/ForceCapture_Cnt&gt;\r\n    &lt;PostAuth_Cnt&gt;0&lt;/PostAuth_Cnt&gt;\r\n    &lt;Return_Cnt&gt;0&lt;/Return_Cnt&gt;\r\n    &lt;Sale_Cnt&gt;13&lt;/Sale_Cnt&gt;\r\n    &lt;Receipt_Cnt&gt;0&lt;/Receipt_Cnt&gt;\r\n    &lt;RepeatSale_Cnt&gt;0&lt;/RepeatSale_Cnt&gt;\r\n    &lt;Activate_Cnt&gt;0&lt;/Activate_Cnt&gt;\r\n    &lt;Deactivate_Cnt&gt;0&lt;/Deactivate_Cnt&gt;\r\n    &lt;Reload_Cnt&gt;0&lt;/Reload_Cnt&gt;\r\n    &lt;Cnt&gt;13&lt;/Cnt&gt;\r\n  &lt;/PaymentMethod&gt;\r\n&lt;/CardTrxSummary&gt;</string>"

我认为,这是我问题的一部分?

修改#2

这就是我最终要做的就是让它发挥作用。我确信有更好的方法:

  var tst2 = tst.InnerText.Replace("&lt;", "<").Replace("&gt;", ">").Replace("\r\n", string.Empty);
  Console.WriteLine("Cnt: " + ReadXmlValue1(tst2, "Cnt"));

和我解析它的方法:

private static void ReadXmlValue1(string xmlDocument)
{
  XDocument xdoc = XDocument.Parse(xmlDocument);
  //XNamespace ns = "http://www.namespaceuri.com/Admin/ws";
  var payments = from p in xdoc.Descendants("PaymentMethod")
                 select new
                 {
                   Sale = (decimal)p.Element("Sale"),
                   SaleCount = (int)p.Element("Sale_Cnt"),
                   PaymentType = (string)p.Element("Payment_Type_ID")
                 };
  Console.WriteLine("Count: " + payments.Count());
  foreach (var payment in payments)
  {
    Console.WriteLine("Sale: " + payment.Sale);
    Console.WriteLine("Sale_Cnt: " + payment.SaleCount);
    Console.WriteLine("Payment_Type_ID: " + payment.PaymentType);
  }

}

修改#3

这就是我创建xmldocument的方式:

/// <summary>
///  Get Data in xml format by url
/// </summary>
/// <param name="url"></param>
/// <returns></returns>
private static XmlDocument GetXmlDataFromUrl(string url)
{
  //requesting the particular web page
  var httpRequest = (HttpWebRequest)WebRequest.Create(url);

  //geting the response from the request url
  var response = (HttpWebResponse)httpRequest.GetResponse();

  //create a stream to hold the contents of the response (in this case it is the contents of the XML file
  var receiveStream = response.GetResponseStream();

  //creating XML document
  var mySourceDoc = new XmlDocument();

  //load the file from the stream
  if (receiveStream != null)
  {
    mySourceDoc.Load(receiveStream);

    //close the stream
    receiveStream.Close();
    return mySourceDoc;
  }
  return null;
}

2 个答案:

答案 0 :(得分:1)

您可以使用LINQ to XML获取强类型匿名付款对象列表:

WebClient client = new WebClient();
string content = client.DownloadString(url);

XDocument xdoc = XDocument.Parse(content);
XNamespace ns = "http://www.namespaceuri.com/Admin/ws";
var payments = from p in xdoc.Descendants(ns + "PaymentMethod")
                select new {
                    Sale = (decimal)p.Element(ns + "Sale"),
                    SaleCount = (int)p.Element(ns + "Sale_Cnt"),
                    PaymentType = (string)p.Element(ns + "Payment_Type_ID")
                };

请记住,您的xml已声明了名称空间,因此您应在指定元素名称时提供它。

用法:

foreach(var payment in payments)
{
   Console.WriteLine("Sale: " + payment.Sale);
   Console.WriteLine("Sale_Cnt: " + payment.SaleCount);
   Console.WriteLine("Payment_Type_ID: " + payment.PaymentType);
}

答案 1 :(得分:0)

XmlNode node = xmlDocument.SelectSingleNode("/string/CardTrxSummary/PaymentMethod");
Console.WriteLine("Sale: " + node.SelectSingleNode("Sale").InnerText);
Console.WriteLine("Sale_Cnt: " + node.SelectSingleNode("Sale_Cnt").InnerText);
Console.WriteLine("Payment_Type_ID: " + node.SelectSingleNode("Payment_Type_ID").InnerText);

或者你可以改用getElementByTagName;

Console.WriteLine("Sale: " + xmlDocument.getElementByTagName("Sale").InnerText);
Console.WriteLine("Sale_Cnt: " + xmlDocument.getElementByTagName("Sale_Cnt").InnerText);
Console.WriteLine("Payment_Type_ID: " + xmlDocument.getElementByTagName("Payment_Type_ID").InnerText);

只是注意,上面的2方法假设标签永远不会返回null。

如果你想处理可能的null节点,你可以这样做。

string text = xmlDocument.getElementByTagName("Sale") != null ? xmlDocument.getElementByTagName("Sale").InnerText : "unidentified";

以上行的格式如下:

var variable = condition ? A : B;

基本上说如果条件为真,则变量等于A,否则变量等于B。