在.net 当中如何XML序列化一个Collection

简介: Collection主要是指像Array, ArrayList, List, Dictionary, HashTable这些数据类型,大家平时用的很多。如果一个类中有一个Collection类型的成员,在对这个类进行XML序列化的时候,应该如何处理?应该说在.net当中这是比较简单的,只要建立一个XmlSerializer类就可以帮你自动搞定,不过有的时候你可能需要对自动的序列化过程施加更多的控制,比如XML的结构是实现固定的,你必须按照要求去生成XML结构。

Collection主要是指像Array, ArrayList, List, Dictionary, HashTable这些数据类型,大家平时用的很多。如果一个类中有一个Collection类型的成员,在对这个类进行XML序列化的时候,应该如何处理?应该说在.net当中这是比较简单的,只要建立一个XmlSerializer类就可以帮你自动搞定,不过有的时候你可能需要对自动的序列化过程施加更多的控制,比如XML的结构是实现固定的,你必须按照要求去生成XML结构。

使用不同的属性可以灵活的控制生成的XML,这里我就不多介绍了,主要讲一下如何序列化比较复杂的Collection结构。下面的方法,对于所有实现了IEnumerable接口的Collection都有效。

我使用MSDN中的例子,不过没有使用数组或者ArrayList,而是使用了比较高级的数据类型List<T>,希望在讲解如何序列化XML的同时给使用List<T>的同学提供点参考。

序列化一个List<T>

下面的代码示范了如何序列化一个List<T>,实际上和序列化其它类一样,把这个类扔给Serialize()函数即可。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Xml.Serialization;
using System.IO;
 
namespace SerializeCollection
{
    class Program
    {
        static void Main(string[] args)
        {
            Program test = new Program();
            test.SerializeDocument("e:\\books.xml");
        }
 
        public void SerializeDocument(string filename)
        {
            // Creates a new XmlSerializer.
            XmlSerializer s =
            new XmlSerializer(typeof(MyRootClass));
 
            // Writing the file requires a StreamWriter.
            TextWriter myWriter = new StreamWriter(filename);
 
            // Creates an instance of the class to serialize. 
            MyRootClass myRootClass = new MyRootClass();
            
            //create items
            Item item1 = new Item();
            // Sets the objects' properties.
            item1.ItemName = "Widget1";
            item1.ItemCode = "w1";
            item1.ItemPrice = 231;
            item1.ItemQuantity = 3;
 
            
            Item item2 = new Item();
            // Sets the objects' properties.
            item2.ItemName = "Widget2";
            item2.ItemCode = "w2";
            item2.ItemPrice = 800;
            item2.ItemQuantity = 2;
 
            // Sets the class's Items property to the list.
            myRootClass.Items.Add(item1);
            myRootClass.Items.Add(item2);
 
            /* Serializes the class, writes it to disk, and closes 
               the TextWriter. */
            s.Serialize(myWriter, myRootClass);
            myWriter.Close();
        }
 
    }
 
    // This is the class that will be serialized.
    [Serializable]
    public class MyRootClass
    {
        public MyRootClass()
        {
            items = new List<Item>();
        }
 
        private List<Item> items;
 
        public List<Item> Items
        {
            get { return items; }
            set { items = value; }
        }
    }
 
    public class Item
    {
        [XmlElement(ElementName = "OrderItem")]
        public string ItemName;
        public string ItemCode;
        public decimal ItemPrice;
        public int ItemQuantity;
    }
 
 
}

最后序列化成的XML:

<?xml version="1.0" encoding="utf-8"?>
<MyRootClass xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
  <Items>
    <Item>
      <OrderItem>Widget1</OrderItem>
      <ItemCode>w1</ItemCode>
      <ItemPrice>231</ItemPrice>
      <ItemQuantity>3</ItemQuantity>
    </Item>
    <Item>
      <OrderItem>Widget2</OrderItem>
      <ItemCode>w2</ItemCode>
      <ItemPrice>800</ItemPrice>
      <ItemQuantity>2</ItemQuantity>
    </Item>
  </Items>
</MyRootClass>

如果这个List<T>中的成员的类还有继承关系

现在把情况变得复杂一点,因为多态,List<T>中的类可能是指定类型的子类型,这个时候会出现什么情况呢?

我们增加一个BookItem类,继承自Item 类。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Xml.Serialization;
using System.IO;
 
namespace SerializeCollection
{
    class Program
    {
        static void Main(string[] args)
        {
            Program test = new Program();
            test.SerializeDocument("e:\\books.xml");
        }
 
        public void SerializeDocument(string filename)
        {
            // Creates a new XmlSerializer.
            XmlSerializer s =
            new XmlSerializer(typeof(MyRootClass));
 
            // Writing the file requires a StreamWriter.
            TextWriter myWriter = new StreamWriter(filename);
 
            // Creates an instance of the class to serialize. 
            MyRootClass myRootClass = new MyRootClass();
 
            /* Uses a more advanced method of creating an list:
         create instances of the Item and BookItem, where BookItem 
         is derived from Item. */
            Item item1 = new Item();
            // Sets the objects' properties.
            item1.ItemName = "Widget1";
            item1.ItemCode = "w1";
            item1.ItemPrice = 231;
            item1.ItemQuantity = 3;
 
            BookItem bookItem = new BookItem();
            // Sets the objects' properties.
            bookItem.ItemCode = "w2";
            bookItem.ItemPrice = 123;
            bookItem.ItemQuantity = 7;
            bookItem.ISBN = "34982333";
            bookItem.Title = "Book of Widgets";
            bookItem.Author = "John Smith";
 
            // Sets the class's Items property to the list.
            myRootClass.Items.Add(item1);
            myRootClass.Items.Add(bookItem);
 
            /* Serializes the class, writes it to disk, and closes 
               the TextWriter. */
            s.Serialize(myWriter, myRootClass);
            myWriter.Close();
        }
 
    }
 
    // This is the class that will be serialized.
    [Serializable]
    public class MyRootClass
    {
        public MyRootClass()
        {
            items = new List<Item>();
        }
 
        private List<Item> items;
 
        public List<Item> Items
        {
            get { return items; }
            set { items = value; }
        }
    }
 
    public class Item
    {
        [XmlElement(ElementName = "OrderItem")]
        public string ItemName;
        public string ItemCode;
        public decimal ItemPrice;
        public int ItemQuantity;
    }
 
    
 
 
 
 
 
 
 
 
 
 
 
 
 
}

修改代码后,我们再运行,出现如下错误“不应是类型 SerializeCollection.BookItem。使用 XmlInclude 或 SoapInclude 属性静态指定非已知的类型”,看来是系统在做序列化的时候,搞不清楚List中的成员到底是什么类型。这个时候就要使用XmlArrayItem来帮忙了。MyRootClass类的Item成员前加入XmlArrayItem标志。

[XmlArrayItem(ElementName= "Item", 
   IsNullable=true,
   Type = typeof(Item),
   Namespace = "http://www.aboutdnn.com"),
   XmlArrayItem(ElementName = "BookItem", 
   IsNullable = true, 
   Type = typeof(BookItem),
   Namespace = http://www.aboutdnn.com)]

修改后的代码如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Xml.Serialization;
using System.IO;
 
namespace SerializeCollection
{
    class Program
    {
        static void Main(string[] args)
        {
            Program test = new Program();
            test.SerializeDocument("e:\\books.xml");
        }
 
        public void SerializeDocument(string filename)
        {
            // Creates a new XmlSerializer.
            XmlSerializer s =
            new XmlSerializer(typeof(MyRootClass));
 
            // Writing the file requires a StreamWriter.
            TextWriter myWriter = new StreamWriter(filename);
 
            // Creates an instance of the class to serialize. 
            MyRootClass myRootClass = new MyRootClass();
 
            /* Uses a more advanced method of creating an list:
         create instances of the Item and BookItem, where BookItem 
         is derived from Item. */
            Item item1 = new Item();
            // Sets the objects' properties.
            item1.ItemName = "Widget1";
            item1.ItemCode = "w1";
            item1.ItemPrice = 231;
            item1.ItemQuantity = 3;
 
            BookItem bookItem = new BookItem();
            // Sets the objects' properties.
            bookItem.ItemCode = "w2";
            bookItem.ItemPrice = 123;
            bookItem.ItemQuantity = 7;
            bookItem.ISBN = "34982333";
            bookItem.Title = "Book of Widgets";
            bookItem.Author = "John Smith";
 
            // Sets the class's Items property to the list.
            myRootClass.Items.Add(item1);
            myRootClass.Items.Add(bookItem);
 
            /* Serializes the class, writes it to disk, and closes 
               the TextWriter. */
            s.Serialize(myWriter, myRootClass);
            myWriter.Close();
        }
 
    }
 
    // This is the class that will be serialized.
    [Serializable]
    public class MyRootClass
    {
        public MyRootClass()
        {
            items = new List<Item>();
        }
 
        private List<Item> items;
 
        
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
        public List<Item> Items
        {
            get { return items; }
            set { items = value; }
        }
    }
 
    public class Item
    {
        [XmlElement(ElementName = "OrderItem")]
        public string ItemName;
        public string ItemCode;
        public decimal ItemPrice;
        public int ItemQuantity;
    }
 
    public class BookItem : Item
    {
        public string Title;
        public string Author;
        public string ISBN;
    }
 
 
 
}

序列化后的XML如下:

 

<?xml version="1.0" encoding="utf-8"?>
<MyRootClass xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
  <Items>
    <Item xmlns="http://www.aboutdnn.com">
      <OrderItem>Widget1</OrderItem>
      <ItemCode>w1</ItemCode>
      <ItemPrice>231</ItemPrice>
      <ItemQuantity>3</ItemQuantity>
    </Item>
    <BookItem xmlns="http://www.aboutdnn.com">
      <ItemCode>w2</ItemCode>
      <ItemPrice>123</ItemPrice>
      <ItemQuantity>7</ItemQuantity>
      <Title>Book of Widgets</Title>
      <Author>John Smith</Author>
      <ISBN>34982333</ISBN>
    </BookItem>
  </Items>
</MyRootClass>

可以看到,已经根据不同的数据类型,序列化为不同名字的节点。这个时候,如果你还想修改XML中<Items>节点的名字或者添加属性,XmlArrayAttribute可以帮你的忙,这个你可以自己试试。

参考文档:

XmlArrayAttribute Class:http://msdn.microsoft.com/en-us/library/system.xml.serialization.xmlarrayattribute.aspx

对于所有控制XML序列化的Attributes,请参考这里:Attributes That Control XML Serialization

目录
相关文章
|
XML 存储 C#
C#三十 Ado.net和XML
C#三十 Ado.net和XML
60 0
|
XML 存储 JSON
使用自定义XML配置文件在.NET桌面程序中保存设置
本文将详细介绍如何在.NET桌面程序中使用自定义的XML配置文件来保存和读取设置。除了XML之外,我们还将探讨其他常见的配置文件格式,如JSON、INI和YAML,以及它们的优缺点和相关的NuGet类库。最后,我们将重点介绍我们为何选择XML作为配置文件格式,并展示一个实用的示例。
131 0
|
6月前
|
XML 开发框架 .NET
C# .NET面试系列八:ADO.NET、XML、HTTP、AJAX、WebService
## 第二部分:ADO.NET、XML、HTTP、AJAX、WebService #### 1. .NET 和 C# 有什么区别? .NET(通用语言运行时): ```c# 定义:.NET 是一个软件开发框架,提供了一个通用的运行时环境,用于在不同的编程语言中执行代码。 作用:它为多语言支持提供了一个统一的平台,允许不同的语言共享类库和其他资源。.NET 包括 Common Language Runtime (CLR)、基础类库(BCL)和其他工具。 ``` C#(C Sharp): ```c# 定义: C# 是一种由微软设计的面向对象的编程语言,专门为.NET 平台开发而创建。 作
310 2
|
6月前
|
XML 前端开发 定位技术
C#(NET Core3.1 MVC)生成站点地图(sitemap.xml)
C#(NET Core3.1 MVC)生成站点地图(sitemap.xml)
98 0
|
XML 数据采集 JavaScript
基于.Net开源Html解析器,此外还支持SVG、XML等格式
基于.Net开源Html解析器,此外还支持SVG、XML等格式
64 0
|
存储 JSON JavaScript
.NET MVC第九章、Web Api Json序列化与反序列化
.NET MVC第九章、Web Api Json序列化与反序列化
209 0
.NET MVC第九章、Web Api Json序列化与反序列化
|
XML JSON 数据格式
.NET中XML序列化和反序列化常用类和用来控制XML序列化的属性总结(XmlSerializer,XmlTypeAttribute,XmlElementAttribute,XmlAttributeAttribute,XmlArrayAttribute...)
.NET中XML序列化和反序列化常用类和用来控制XML序列化的属性总结(XmlSerializer,XmlTypeAttribute,XmlElementAttribute,XmlAttributeAttribute,XmlArrayAttribute...)
282 0
|
JSON 开发框架 算法
.Net 序列化枚举为字符串
我所做的项目是需要调用业务算法的,算法中有一个入参是油品的性质,这个性质有名称、编码、类型等属性,其中类型是固定质量性质、体积性质和其他性质这三种,所以我把其作为枚举类型。问题也由此产生,默认情况下,枚举是以其整数形式进行 JSON 序列化,这就需要同研发算法的同事约定好数值的含义。但是经过协商,算法同事要求我们传递成字符串。因此,我们希望它们在一些情况下以字符串的形式进行序列化。本文将讲解实现这一目标的各种方法。
|
XML 开发框架 .NET
【原创】ASP.NET WebApi接收xml文件 xml序列化
如何新建WebApi项目就不用我介绍了吧。 直接进入正题。 首先,在.net平台不论要接收什么,肯定是从Request里获取。 大家肯定对普通的参数获取非常熟悉了,下面就介绍一下如何从Request获取文件流. System.IO.Stream sm =Request.Content.ReadAsStreamAsync().Result 这个方法返回的是一个System.IO.Stream类型,如果是单纯想获得文件,到这一步你就可以把它保存到本地了。
9263 0
|
XML JSON 数据格式
[.NET]使用十年股价对比各种序列化技术
原文:[.NET]使用十年股价对比各种序列化技术 1. 前言 上一家公司有搞股票,当时很任性地直接从服务器读取一个股票10年份的股价(还有各种指标)在客户端的图表上显示,而且因为是桌面客户端,传输的数据也是简单粗暴地使用Soap序列化。
858 0