使用C#中的XMLReader解析具有多个类似节点的XML

3

我收到了一个SOAP服务的响应。它看起来像这样:

<ArrayOfPaymentHistory_ST xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns="http://tempuri.org/">
  <PaymentHistory_ST>
    <tTRANSDATE>2015-01-05T08:58:17</tTRANSDATE>
    <tPAIDTO>2015-02-01T00:00:00</tPAIDTO>
    <dAMOUNT>110</dAMOUNT>
    <dBALANCE>0</dBALANCE>
    <TRANSKIND>Auto CC Payment</TRANSKIND>
    <csUnit>Lee</csUnit>
  </PaymentHistory_ST>
  <PaymentHistory_ST>
    <tTRANSDATE>2015-01-01T08:58:17</tTRANSDATE>
    <tPAIDTO>2015-01-01T00:00:00</tPAIDTO>
    <dAMOUNT>-110</dAMOUNT>
    <dBALANCE>110</dBALANCE>
    <TRANSKIND>Rent Posted</TRANSKIND>
    <csUnit>Lee</csUnit>
  </PaymentHistory_ST>
</ArrayOfPaymentHistory_ST>

服务响应


我定义了一个模型,如下所示:

public class payment_history
{
    public DateTime tTRANSDATE { get; set; }
    public DateTime tPAIDTO { get; set; }
    public double dAMOUNT { get; set; }
    public double dBALANCE { get; set; }
    public string TRANSKIND { get; set; }
}

我正在尝试解析这个XML响应,并创建一个payment_history类型的对象列表。然后,我将把这个对象发送到我的标记上,并在表格中显示。我发现很多关于使用XMLReader进行XML解析的帖子。但是我仍然无法找到一个能够优雅地解决这个问题的好方法。
这是我的C#代码。
List<payment_history> ph = new List<payment_history>();
payment_history p = new payment_history();

            using (WebResponse response = request.GetResponse())
            {
                using (StreamReader rd = new StreamReader(response.GetResponseStream()))
                {
                    string soapResult = rd.ReadToEnd();

                    using (XmlReader reader = XmlReader.Create(new StringReader(soapResult)))
                    {
                        #region parseXML
                        while (reader.Read())
                        {
                            foreach (var el in reader.ReadOuterXml())
                            {
                                if (reader.IsStartElement())
                                {
                                    //return only when you have START tag
                                    switch (reader.Name.ToString())
                                    {
                                        case "tTRANSDATE":
                                            p.tTRANSDATE = Convert.ToDateTime(reader.ReadString());
                                            break;

                                        case "tPAIDTO":
                                            p.tPAIDTO = Convert.ToDateTime(reader.ReadString());
                                            break;

                                        case "dAMOUNT":
                                            p.dAMOUNT = Convert.ToDouble(reader.ReadString());
                                            break;

                                        case "dBALANCE":
                                            p.dBALANCE = Convert.ToDouble(reader.ReadString());
                                            break;
                                        case "TRANSKIND":
                                            p.TRANSKIND = reader.ReadString();
                                            break;
                                    }
                                    ph.Add(p);
                                }
                            }
                        }
                        #endregion
                    }
                }
            }
            return ph;

有人能指导我如何使这个工作吗?我是c#的新手。任何形式的帮助将不胜感激。


我尝试了遵循LINQ2XML,但没有返回任何东西!!

ph = XDocument.Parse(soapResult)
                 .Descendants("PaymentHistory_ST")
                 .Select(g => new payment_history
                 {
                     tTRANSDATE = (DateTime)g.Element("tTRANSDATE"),
                     tPAIDTO = (DateTime)g.Element("tPAIDTO"),
                     dAMOUNT = (double)g.Element("dAMOUNT"),
                     dBALANCE = (double)g.Element("dBALANCE"),
                     TRANSKIND = (string)g.Element("TRANSKIND")
                 }).ToList();

可能LINQ2XML会是一个不错的解决方案,但我仍然不知道如何使用它!!有人能给些建议吗?! - vohrahul
1个回答

2

使用

XDocument doc = XDocument.Load(response.GetResponseStream());
XNamespace df = "http://tempuri.org/";

List<payment_history> ph = doc.Descendants(df + "PaymentHistory_ST")
                 .Select(g => new payment_history
                 {
                     tTRANSDATE = (DateTime)g.Element(df + "tTRANSDATE"),
                     tPAIDTO = (DateTime)g.Element(df + "tPAIDTO"),
                     dAMOUNT = (double)g.Element(df + "dAMOUNT"),
                     dBALANCE = (double)g.Element(df + "dBALANCE"),
                     TRANSKIND = (string)g.Element(df + "TRANSKIND")
                 }).ToList();

我尝试过了,但它不起作用。奇怪的是它没有给出任何错误,同时“ph”的计数为0。我验证了该服务返回了许多有效项目的响应。 - vohrahul
我添加了一张截图供您参考。请指教。 - vohrahul
你发布的XML是否还有更多内容?根元素标记是 <ArrayOfPaymentHistory_ST xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns="http://tempuri.org/">,就像你的示例一样吗?还是有更多包含你发布的示例的元素? - Martin Honnen
在你的截屏中有更多元素,因此我的代码无法按照预期读取包含数据目标命名空间的根元素的命名空间,而在你的截屏中它是不同的元素。因此我现在硬编码了命名空间。 - Martin Honnen
你能解释一下为什么我们需要添加命名空间吗?为什么你要在所有节点名称前面添加 df? - vohrahul
1
使用<root xmlns="http://example.com/"><child><grand-child/></child></root>的标记,所有三个XML元素都在根元素上声明的命名空间中。为了使用LINQ to XML选择这样的元素,我们需要构建一个XName,将XNamespace和字符串连接起来。有关详细信息,请参见http://msdn.microsoft.com/en-us/library/bb387093%28v=vs.110%29.aspx。 - Martin Honnen

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接