使用Servicestack.Text进行XML反序列化

Bhu*_*ake 9 c# xml-deserialization servicestack

我正在学习Servicestack.Text库,因为它具有一些最好的功能.我正在尝试将XML反序列化为我的一个DTO,如下所示;

C#代码: [与控制台应用程序相关的代码]

class Program
    {
        static void Main(string[] args)
        {
            string str = "http://static.cricinfo.com/rss/livescores.xml";
            WebClient w = new WebClient();
            string xml = w.DownloadString(str);
            Response rss = xml.FromXml<Response>();
            foreach (var item in rss.rss.Channel.item)
            {
                Console.WriteLine(item.title);
            }
            Console.Read();
        }
    }
Run Code Online (Sandbox Code Playgroud)

您可以在str[程序中的给定]中查看XML文件.我为反序列化准备了DTO.它们如下:

public  class Response
{
   public RSS rss { get; set; }
}

public class RSS
{
   public string Version { get; set; }
   public ChannelClass Channel { get; set; }
}

public class ChannelClass
{
   public string title { get; set; }
   public string ttl { get; set; }
   public string description { get; set; }
   public string link { get; set; }
   public string copyright { get; set; }
   public string language { get; set; }
   public string pubDate { get; set; }
   public List<ItemClass> item { get; set; }
}

public class ItemClass
{
   public string title { get; set; }
   public string link { get; set; }
   public string description { get; set; }
   public string guid { get; set; }
}
Run Code Online (Sandbox Code Playgroud)

当我运行程序时,我得到一个例外,如下所示:

在此输入图像描述

所以,要改变Elementnamespace,我做了以下解决方法:

我把DataContractAttribute我的Response课程放在下面:

[DataContract(Namespace = "")]
public  class Response
{
   public RSS rss { get; set; }
}
Run Code Online (Sandbox Code Playgroud)

Element在反序列化之前添加了以下两行来更改名称,如下所示

  //To change rss Element to Response as in Exception
  xml = xml.Replace("<rss version=\"2.0\">","<Response>");
  //For closing tag
  xml = xml.Replace("</rss>","</Response>");
Run Code Online (Sandbox Code Playgroud)

但是,它在foreach循环中给出了另一个例外,因为反序列化的rss对象是null.那么,我应该如何使用适当的方式对其进行反序列化Servicestack.Text

注意 :

我很清楚如何使用其他库进行反序列化,我只想使用ServiceStack.

Eam*_*nne 23

TLDR:用于XmlSerializer从无法控制的xml方言反序列化; ServiceStack是专为代码优先开发而设计的,不能适用于通用的xml解析.


ServiceStack.Text 没有实现自定义的Xml序列化程序 - 它DataContractSerializer在底层使用. FromXml仅仅是语法糖.

使用DataContractSerializer解析XML

正如您所注意到的那样,DataContractSerializer命名空间很挑剔.一种方法是在类上明确指定命名空间,但是如果这样做,则需要指定[DataMember]所有位置,因为它假定如果有任何内容是明确的,那么一切都是.您可以使用程序集级属性(例如在AssemblyInfo.cs中)来解决此问题,以声明默认命名空间:

[assembly: ContractNamespace("", ClrNamespace = "My.Namespace.Here")]
Run Code Online (Sandbox Code Playgroud)

这解决了命名空间问题.

但是,您无法解决DataContractSerializer的其他2个问题:

  • 它不会使用属性(在您的情况下version)
  • 它要求集合如同item包装名称和元素名称(类似于项目和项目)

您无法解决这些限制,因为DataContractSerializer它不是通用的XML解析器.它旨在轻松生成和使用API​​,而不是将任意XML映射到.NET数据结构.你永远不会得到它来解析rss; 因此,ServiceStack.Text(它只包装它)也无法解析它.

相反,使用XmlSerializer.

运用 XmlSerializer

这是相当直接的.您可以使用以下内容解析输入:

var serializer = new XmlSerializer(typeof(RSS));
RSS rss = (RSS)serializer.Deserialize(myXmlReaderHere);
Run Code Online (Sandbox Code Playgroud)

诀窍是注释各个字段,使它们与你的xml方言相匹配.例如,在您的情况下,将是:

[XmlRoot("rss")]
public class RSS
{
    [XmlAttribute]
    public string version { get; set; }
    public ChannelClass channel { get; set; }
}

public class ChannelClass
{
    public string title { get; set; }
    public string ttl { get; set; }
    public string description { get; set; }
    public string link { get; set; }
    public string copyright { get; set; }
    public string language { get; set; }
    public string pubDate { get; set; }
    [XmlElement]
    public List<ItemClass> item { get; set; }
}

public class ItemClass
{
    public string title { get; set; }
    public string link { get; set; }
    public string description { get; set; }
    public string guid { get; set; }
}
Run Code Online (Sandbox Code Playgroud)

因此,一些明智的属性足以让它根据需要解析XML.

总结:您不能使用ServiceStack,因为它使用DataContractSerializer.ServiceStack/DataContractSerializer是为您控制架构的场景而设计的. XmlSerializer改用.