XmlSerializer序列化

  XmlSerializer在命名空间using System.Xml.Serialization下。

序列化和反序列化的代码:

using System.IO;
using System.Xml;
using System.Xml.Serialization;
namespace ConsoleApplication1
{
    public class PublicFunction
    {
        //序列化到XmlNode
        public static void Serialize<T>(T t, out XmlNode node)
        {
            XmlDocument doc = new XmlDocument();
            using (MemoryStream ms = new MemoryStream())
            {
                XmlSerializer serializer = new XmlSerializer(typeof(T), typeof(T).Assembly.GetName().Name);
                serializer.Serialize(ms, t);
                ms.Position = 0;
                doc.Load(ms);
                node = doc.LastChild;
            }
        }
        //反序列化XmlNode中的数据
        public static void DeSerialize<T>(XmlNode node, out T t)
        {
            XmlSerializer serializer = new XmlSerializer(typeof(T), typeof(T).Assembly.GetName().Name);
            XmlNodeReader reader = new XmlNodeReader(node);
            t=(T)serializer.Deserialize(reader);
        }

        //序列化到文件
        public static void SerializeFile<T>(T t, string filepath)
        {
           using(XmlWriter writer = new XmlTextWriter(filepath, Encoding.UTF8))
           {
               XmlSerializer serializer = new XmlSerializer(typeof(T));
               serializer.Serialize(writer,t);
           }
        }

        //将文件中的内容反序列化
        public static void DeSerializeFile<T>(string Filepath,out T t)
        {
            t = default(T);
            using (XmlReader reader = new XmlTextReader(Filepath))
            {
                XmlSerializer serializer = new XmlSerializer(typeof(T));
                serializer.Deserialize(reader);
            }
        }
    }
}

  XmlSerializer序列化规则:

   1.只能序列化public成员变量,protected、private、public static 声明的字段都不支持序列化。

   2.对象属性必须是可读可写。

   3.对象必须要有一个无参构造函数,用于反序列化。

   4.序列化的顺序和字段定义的顺序保持一致。

   5.序列化子类,而声明的是基类时,需要在基类中使用XmlInclude声明子类类型,例如下面的listCCAbstract字段。

using System.Xml;
using System.Xml.Serialization;
using System.IO;
namespace ConsoleApplication1
{
    class Program
    {
        static void Main(string[] args)
        {
            Aa a = new Aa();
            PublicFunction.SerializeFile<Aa>(a, @"D:\1.txt");
        }
    }
    [XmlRoot("AaRoot")]
    public class Aa 
    {
        public static string StaticParam;
        [XmlAttribute("Name")]
        public string name;

        [XmlIgnore]
        public string Ignore;
        protected string proparam;
        private int age;
        public int pAttribute
        {
            get;
            set;
        }
        [XmlArray("List")]
        [XmlArrayItem("Item")]
        public List<CCAbstract> listCCAbstract { get; set; }

        public Aa()
        {
            StaticParam = "StaticParam";
            this.name = "Aa";
            this.proparam = "Praparam";
            this.age = 23;
            pAttribute = 23;
            Ignore = "Ignpore";
            listCCAbstract = new List<CCAbstract> { new Cc("testCcAbstract") };
        }
    }

    [XmlInclude(typeof(Cc))]
    public abstract class CCAbstract
    {
    }

    public class Cc:CCAbstract
    {
        public string CName;
        public Cc()
        {
        }
        public Cc(string name)
        {
          this.CName=name;
        }
    }
}

 序列化后的xml内容为:

<?xml version="1.0" encoding="utf-8"?>
<AaRoot xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
        xmlns:xsd="http://www.w3.org/2001/XMLSchema" 
        Name="Aa">
  <pAttribute>23</pAttribute>
  <List>
    <Item xsi:type="Cc">
      <CName>testCcAbstract</CName>
    </Item>
  </List>
</AaRoot>

 可以看到XmlRoot可以指定根结点名称,name不再作为XmlElement而是变成了AaRoot的属性。XmlIgnore标签标识的字段不进行序列化,protected、private、public static 声明的字段都不支持序列化。XmlArray可以指定List类型序列化后的名称,XmlArrayItem则对应List中的元素。

 

   6.尽量使用XmlSerializer(Type)、XmlSerializer(Type,String)这两种构造函数,如果使用其它构造函数则会导致内存泄漏。

详情请看

https://msdn.microsoft.com/zh-cn/library/system.xml.serialization.xmlserializer(VS.80).aspx

为了提高性能,XML 序列化基础结构动态生成程序集,以便对指定类型进行序列化和反序列化。该基础结构将找到并重新使用这些程序集。仅当使用以下构造函数时,才会发生此行为:

System.Xml.Serialization.XmlSerializer(Type)

System.Xml.Serialization.XmlSerializer(Type,String)

如果使用任何其他构造函数,则将生成同一个程序集的多个版本,这些版本始终不予卸载,从而导致内存泄漏和性能低下。最简单的解决方案是使用上面两个构造函数中的一个。

 

posted @ 2015-04-26 14:21  lh_cn  阅读(808)  评论(0编辑  收藏  举报