C# 序列化理解 1(转)

    序列化又称串行化,是.NET运行时环境用来支持用户定义类型的流化的机制。其目的是以某种存储形成使自定义对象持久化,或者将这种对象从一个地方传输到另一个地方。
    .NET框架提供了两种串行化的方式:1、是使用BinaryFormatter进行串行化;2、使用SoapFormatter进行串行化;3、使用XmlSerializer进行串行化。第一种方式提供了一个简单的二进制数据流以及某些附加的类型信息,而第二种将数据流格式化为XML存储;第三种其实和第二种差不多也是XML的格式存储,只不过比第二种的XML格式要简化很多(去掉了SOAP特有的额外信息)。
    可以使用[Serializable]属性将类标志为可序列化的。如果某个类的元素不想被序列化,1、2可以使用[NonSerialized]属性来标志,2、可以使用[XmlIgnore]来标志。
    1、使用BinaryFormatter进行串行化
    下面是一个可串行化的类:
    

C# 序列化理解 1(转)using System;
C# 序列化理解 1(转)using System.Data;
C# 序列化理解 1(转)using System.Configuration;
C# 序列化理解 1(转)using System.Web;
C# 序列化理解 1(转)using System.Web.Security;
C# 序列化理解 1(转)using System.Web.UI;
C# 序列化理解 1(转)using System.Web.UI.WebControls;
C# 序列化理解 1(转)using System.Web.UI.WebControls.WebParts;
C# 序列化理解 1(转)using System.Web.UI.HtmlControls;
C# 序列化理解 1(转)using System.IO;
C# 序列化理解 1(转)using
 System.Runtime.Serialization.Formatters.Binary;
C# 序列化理解 1(转)/// <summary>
C# 序列化理解 1(转)/// ClassToSerialize 的摘要说明
C# 序列化理解 1(转)/// </summary>
C# 序列化理解 1(转)[Serializable]
C# 序列化理解 1(转)public class ClassToSerialize
C# 序列化理解 1(转){
C# 序列化理解 1(转)    public int id = 100;
C# 序列化理解 1(转)    public string name = "Name";
C# 序列化理解 1(转)    [NonSerialized]
C# 序列化理解 1(转)    public string Sex = "男";
C# 序列化理解 1(转)}
C# 序列化理解 1(转)


    下面是串行化和反串行化的方法:
    

C# 序列化理解 1(转)public void SerializeNow()
C# 序列化理解 1(转)    {
C# 序列化理解 1(转)        ClassToSerialize c = new ClassToSerialize();
C# 序列化理解 1(转)        FileStream fileStream = new FileStream("c:\\temp.dat", FileMode.Create);
C# 序列化理解 1(转)        BinaryFormatter b = new BinaryFormatter();
C# 序列化理解 1(转)        b.Serialize(fileStream, c);
C# 序列化理解 1(转)        fileStream.Close();
C# 序列化理解 1(转)    }
C# 序列化理解 1(转)    public void DeSerializeNow()
C# 序列化理解 1(转)    {
C# 序列化理解 1(转)        ClassToSerialize c = new ClassToSerialize();
C# 序列化理解 1(转)        c.Sex = "kkkk";
C# 序列化理解 1(转)        FileStream fileStream = new FileStream("c:\\temp.dat", FileMode.Open, FileAccess.Read, FileShare.Read);
C# 序列化理解 1(转)        BinaryFormatter b = new BinaryFormatter();
C# 序列化理解 1(转)        c = b.Deserialize(fileStream) as ClassToSerialize;
          Response.Write(c.name);
C# 序列化理解 1(转)        Response.Write(c.Sex);
C# 序列化理解 1(转)        fileStream.Close();
C# 序列化理解 1(转)    }


    调用上述两个方法就可以看到串行化的结果:Sex属性因为被标志为[NonSerialized],故其值总是为null。
    2、使用SoapFormatter进行串行化
    和BinaryFormatter类似,我们只需要做一下简单修改即可:
    a.将using语句中的.Formatter.Binary改为.Formatter.Soap;
    b.将所有的BinaryFormatter替换为SoapFormatter.
    c.确保报存文件的扩展名为.xml
    经过上面简单改动,即可实现SoapFormatter的串行化,这时候产生的文件就是一个xml格式的文件。
    3、使用XmlSerializer进行串行化
    关于格式化器还有一个问题,假设我们需要XML,但是不想要SOAP特有的额外信息,那么我们应该怎么办呢?有两中方案:要么编写一个实现IFormatter接口的类,采用的方式类似于SoapFormatter类,但是没有你不需要的信息;要么使用库类XmlSerializer,这个类不使用Serializable属性,但是它提供了类似的功能。
    如果我们不想使用主流的串行化机制,而想使用XmlSeralizer进行串行化我们需要做一下修改:
    a.添加System.Xml.Serialization命名空间。
    b.Serializable和NoSerialized属性将被忽略,而是使用XmlIgnore属性,它的行为与NoSerialized类似。
    c.XmlSeralizer要求类有个默认的构造器,这个条件可能已经满足了。
    下面看示例:
    要序列化的类:
    

C# 序列化理解 1(转)using System;
C# 序列化理解 1(转)using System.Data;
C# 序列化理解 1(转)using System.Configuration;
C# 序列化理解 1(转)using System.Web;
C# 序列化理解 1(转)using System.Web.Security;
C# 序列化理解 1(转)using System.Web.UI;
C# 序列化理解 1(转)using System.Web.UI.WebControls;
C# 序列化理解 1(转)using System.Web.UI.WebControls.WebParts;
C# 序列化理解 1(转)using System.Web.UI.HtmlControls;
C# 序列化理解 1(转)using System.Xml.Serialization;
C# 序列化理解 1(转)[Serializable]
C# 序列化理解 1(转)public class Person
C# 序列化理解 1(转){
C# 序列化理解 1(转)    private string name;
C# 序列化理解 1(转)    public string Name
C# 序列化理解 1(转)    {
C# 序列化理解 1(转)        get
C# 序列化理解 1(转)        {
C# 序列化理解 1(转)            return name;
C# 序列化理解 1(转)        }
C# 序列化理解 1(转)        set
C# 序列化理解 1(转)        {
C# 序列化理解 1(转)            name = value;
C# 序列化理解 1(转)        }
C# 序列化理解 1(转)    }
C# 序列化理解 1(转)
C# 序列化理解 1(转)
C# 序列化理解 1(转)    public string Sex;
C# 序列化理解 1(转)    public int Age = 31;
C# 序列化理解 1(转)    public Course[] Courses;
C# 序列化理解 1(转)
C# 序列化理解 1(转)    public Person()
C# 序列化理解 1(转)    {
C# 序列化理解 1(转)    }
C# 序列化理解 1(转)    public Person(string Name)
C# 序列化理解 1(转)    {
C# 序列化理解 1(转)        name = Name;
C# 序列化理解 1(转)        Sex = "男";
C# 序列化理解 1(转)    }
C# 序列化理解 1(转)}
C# 序列化理解 1(转)[Serializable]
C# 序列化理解 1(转)public class Course
C# 序列化理解 1(转){
C# 序列化理解 1(转)    public string Name;
C# 序列化理解 1(转)    [XmlIgnore]
C# 序列化理解 1(转)    public string Description;
C# 序列化理解 1(转)    public Course()
C# 序列化理解 1(转)    {
C# 序列化理解 1(转)    }
C# 序列化理解 1(转)    public Course(string name, string description)
C# 序列化理解 1(转)    {
C# 序列化理解 1(转)        Name = name;
C# 序列化理解 1(转)        Description = description;
C# 序列化理解 1(转)    }
C# 序列化理解 1(转)}  
C# 序列化理解 1(转)


    序列化和反序列化方法:

C# 序列化理解 1(转)public void XMLSerialize()
C# 序列化理解 1(转)    {
C# 序列化理解 1(转)        Person c = new Person("cyj");
C# 序列化理解 1(转)        c.Courses = new Course[2];
C# 序列化理解 1(转)        c.Courses[0] = new Course("英语", "交流工具");
C# 序列化理解 1(转)        c.Courses[1] = new Course("数学","自然科学");
C# 序列化理解 1(转)        XmlSerializer xs = new XmlSerializer(typeof(Person));
C# 序列化理解 1(转)        Stream stream = new FileStream("c:\\cyj.XML",FileMode.Create,FileAccess.Write,FileShare.Read);
C# 序列化理解 1(转)        xs.Serialize(stream,c);
C# 序列化理解 1(转)        stream.Close();
C# 序列化理解 1(转)    }
C# 序列化理解 1(转)    public void XMLDeserialize()
C# 序列化理解 1(转)    {
C# 序列化理解 1(转)        XmlSerializer xs = new XmlSerializer(typeof(Person));
C# 序列化理解 1(转)        Stream stream = new FileStream("C:\\cyj.XML",FileMode.Open,FileAccess.Read,FileShare.Read);
C# 序列化理解 1(转)        Person p = xs.Deserialize(stream) as Person;
C# 序列化理解 1(转)        Response.Write(p.Name);
C# 序列化理解 1(转)        Response.Write(p.Age.ToString());
C# 序列化理解 1(转)        Response.Write(p.Courses[0].Name);
C# 序列化理解 1(转)        Response.Write(p.Courses[0].Description);
C# 序列化理解 1(转)        Response.Write(p.Courses[1].Name);
C# 序列化理解 1(转)        Response.Write(p.Courses[1].Description);
C# 序列化理解 1(转)        stream.Close();
C# 序列化理解 1(转)    }


这里Course类的Description属性值将始终为null,生成的xml文档中也没有该节点,如下图:

C# 序列化理解 1(转)<?xml version="1.0"?>
C# 序列化理解 1(转)<Person xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
C# 序列化理解 1(转)  <Sex>男</Sex>
C# 序列化理解 1(转)  <Age>31</Age>
C# 序列化理解 1(转)  <Courses>
C# 序列化理解 1(转)    <Course>
C# 序列化理解 1(转)      <Name>英语</Name>
C# 序列化理解 1(转)      <Description>交流工具</Description>
C# 序列化理解 1(转)    </Course>
C# 序列化理解 1(转)    <Course>
C# 序列化理解 1(转)      <Name>数学</Name>
C# 序列化理解 1(转)      <Description>自然科学</Description>
C# 序列化理解 1(转)    </Course>
C# 序列化理解 1(转)  </Courses>
C# 序列化理解 1(转)  <Name>cyj</Name>
C# 序列化理解 1(转)</Person>


    4、自定义序列化
    如果你希望让用户对类进行串行化,但是对数据流的组织方式不完全满意,那么可以通过在自定义类中实现接口来自定义串行化行为。这个接口只有一个方法,GetObjectData. 这个方法用于将对类对象进行串行化所需要的数据填进SerializationInfo对象。你使用的格式化器将构造SerializationInfo对象,然后在串行化时调用GetObjectData. 如果类的父类也实现了ISerializable,那么应该调用GetObjectData的父类实现。
    如果你实现了ISerializable,那么还必须提供一个具有特定原型的构造器,这个构造器的参数列表必须与GetObjectData相同。这个构造器应该被声明为私有的或受保护的,以防止粗心的开发人员直接使用它。
    示例如下:
    实现ISerializable的类:
    

C# 序列化理解 1(转)using System;
C# 序列化理解 1(转)using System.Data;
C# 序列化理解 1(转)using System.Configuration;
C# 序列化理解 1(转)using System.Web;
C# 序列化理解 1(转)using System.Web.Security;
C# 序列化理解 1(转)using System.Web.UI;
C# 序列化理解 1(转)using System.Web.UI.WebControls;
C# 序列化理解 1(转)using System.Web.UI.WebControls.WebParts;
C# 序列化理解 1(转)using System.Web.UI.HtmlControls;
C# 序列化理解 1(转)using System.Runtime.Serialization;
C# 序列化理解 1(转)using System.Runtime.Serialization.Formatters.Binary;
C# 序列化理解 1(转)/// <summary>
C# 序列化理解 1(转)/// Employee 的摘要说明
C# 序列化理解 1(转)/// </summary>
C# 序列化理解 1(转)[Serializable]
C# 序列化理解 1(转)public class Employee:ISerializable
C# 序列化理解 1(转){
C# 序列化理解 1(转)    public int EmpId=100;
C# 序列化理解 1(转)    public string EmpName="刘德华";
C# 序列化理解 1(转)    [NonSerialized]
C# 序列化理解 1(转)    public string NoSerialString = "NoSerialString-Test";
C# 序列化理解 1(转)    public Employee()
C# 序列化理解 1(转)    {
C# 序列化理解 1(转)        //
C# 序列化理解 1(转)        // TODO: 在此处添加构造函数逻辑
C# 序列化理解 1(转)        //
C# 序列化理解 1(转)    }
C# 序列化理解 1(转)    private Employee(SerializationInfo info, StreamingContext ctxt)
C# 序列化理解 1(转)    {
C# 序列化理解 1(转)        EmpId = (int)info.GetValue("EmployeeId", typeof(int));
C# 序列化理解 1(转)        EmpName = (String)info.GetValue("EmployeeName",typeof(string));
C# 序列化理解 1(转)        //NoSerialString = (String)info.GetValue("EmployeeString",typeof(string));
C# 序列化理解 1(转)    }
C# 序列化理解 1(转)    public void GetObjectData(SerializationInfo info, StreamingContext ctxt)
C# 序列化理解 1(转)    {
C# 序列化理解 1(转)        info.AddValue("EmployeeId", EmpId);
C# 序列化理解 1(转)        info.AddValue("EmployeeName", EmpName);
C# 序列化理解 1(转)        //info.AddValue("EmployeeString", NoSerialString);
C# 序列化理解 1(转)    }
C# 序列化理解 1(转)}
C# 序列化理解 1(转)


    序列化和反序列化方法:

C# 序列化理解 1(转)public void OtherEmployeeClassTest()
C# 序列化理解 1(转)    {
C# 序列化理解 1(转)        Employee mp = new Employee();
C# 序列化理解 1(转)        mp.EmpId = 10;
C# 序列化理解 1(转)        mp.EmpName = "邱枫";
C# 序列化理解 1(转)        mp.NoSerialString = "你好呀";
C# 序列化理解 1(转)        Stream steam = File.Open("c:\\temp3.dat", FileMode.Create);
C# 序列化理解 1(转)        BinaryFormatter bf = new BinaryFormatter();
C# 序列化理解 1(转)        Response.Write("Writing Employee Info:");
C# 序列化理解 1(转)        bf.Serialize(steam,mp);
C# 序列化理解 1(转)        steam.Close();
C# 序列化理解 1(转)        mp = null;
C# 序列化理解 1(转)        //反序列化
C# 序列化理解 1(转)        Stream steam2 = File.Open("c:\\temp3.dat", FileMode.Open);
C# 序列化理解 1(转)        BinaryFormatter bf2 = new BinaryFormatter();
C# 序列化理解 1(转)        Response.Write("Reading Employee Info:");
C# 序列化理解 1(转)        Employee mp2 = (Employee)bf2.Deserialize(steam2);
C# 序列化理解 1(转)        steam2.Close();
C# 序列化理解 1(转)        Response.Write(mp2.EmpId);
C# 序列化理解 1(转)        Response.Write(mp2.EmpName);
C# 序列化理解 1(转)        Response.Write(mp2.NoSerialString);
C# 序列化理解 1(转)    }

C# 序列化理解 1(转)

上一篇:【百度地图API】除夕夜,大家一起来赶走“夕”——删除标注功能


下一篇:something about WinPE系统