面向对象——序列化与反序列化

简介: 面向对象——序列化与反序列化

概念:序列化是将对象状态转换为可保持或传输的形式的过程。如果一个类所创建的对象,能够被序列化,那么要求必须给这个类加上[Serializable]特性。


1.      使用二进制形式序列化对象


必须添加System.Runtime.Serialization.Formatters.Binary;命名空间


2.      使用soap形式序列化对象


3.      使用xml形式序列化对象


例:


public partial class Form1 : Form


   {

       public Form1()


       {

           InitializeComponent();


       }



       private void button1_Click(object sender, EventArgs e)


       {

           Student st = new Student() { RealName = "aa", Gender = "女" };


           FileStream fs = new FileStream(@"D:stu.txt", FileMode.OpenOrCreate);


           BinaryFormatter bf = new BinaryFormatter();


           bf.Serialize(fs, st);//把对象存放到文件流中。


           fs.Close();


       }


       ///


       /// 反序列化取出数据


       ///


       private void button2_Click(object sender, EventArgs e)


       {

           FileStream fs = new FileStream(@"D:stu.txt", FileMode.Open);


           BinaryFormatter bf = new BinaryFormatter();


           Student st = bf.Deserialize(fs) as Student;//反序列化从文件中取出数据


           if (st != null)


           {

               MessageBox.Show(st.RealName.ToString());


           }


           fs.Close();


       }


   }


   [Serializable]


   public class Student


   {

       public string RealName { get; set; }


       public String Gender { get; set; }


   }



1.对象序列化的介绍

(1).NET支持对象序列化的几种方式


二进制序列化:对象序列化之后是二进制形式的,通过BinaryFormatter类来实现的,这个类位于System.Runtime.Serialization.Formatters.Binary命名空间下。


SOAP序列化:对象序列化之后的结果符合SOAP协议,也就是可以通过SOAP 协议传输,通过System.Runtime.Serialization.Formatters.Soap命名空间下的SoapFormatter类来实现的。


XML序列化:对象序列化之后的结果是XML形式的,通过XmlSerializer 类来实现的,这个类位于System.Xml.Serialization命名空间下。XML序列化不能序列化私有数据。


(2)几种序列化的区别


二进制格式和SOAP格式可序列化一个类型的所有可序列化字段,不管它是公共字段还是私有字段。XML格式仅能序列化公共字段或拥有公共属性的私有字段,未通过属性公开的私有字段将被忽略。


使用二进制格式序列化时,它不仅是将对象的字段数据进行持久化,也持久化每个类型的完全限定名称和定义程序集的完整名称(包括包称、版本、公钥标记、区域性),这些数据使得在进行二进制格式反序列化时亦会进行类型检查。SOAP格式序列化通过使用XML命名空间来持久化原始程序集信息。而XML格式序列化不会保存完整的类型名称或程序集信息。这便利XML数据表现形式更有终端开放性。如果希望尽可能延伸持久化对象图的使用范围时,SOAP格式和XML格式是理想选择。


(3)使用特性对序列化的控制


要让一个对象支持.Net序列化服务,用户必须为每一个关联的类加上[Serializable]特性。如果类中有些成员不适合参与序列化(比如:密码字段),可以在这些域前加上[NonSerialized]特性。


2.使用二进制序列化和反序列化

(1)二进制序列化与反序列化的程序示例


   [Serializable]  //必须添加序列化特性


   public class Person


   {


       private string Name;//姓名


       private bool Sex;//性别,是否是男


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   [Serializable]  //必须添加序列化特性


   public class Programmer : Person


   {


       private string Language;//编程语言


       public Programmer(string name, bool sex, string language) : base(name, sex)


       {


           this.Language = language;


       }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           //创建Programmer列表,并添加对象


           List list = new List();


           list.Add(new Programmer("李志伟", true, "C#"));


           list.Add(new Programmer("Coder2", false, "C++"));


           list.Add(new Programmer("Coder3", true, "Java"));


           //使用二进制序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.dat";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create, FileAccess.ReadWrite);


           BinaryFormatter binFormat = new BinaryFormatter();//创建二进制序列化器


           binFormat.Serialize(fStream, list);


           //使用二进制反序列化对象


           list.Clear();//清空列表


           fStream.Position = 0;//重置流位置


           list = (List)binFormat.Deserialize(fStream);//反序列化对象


           foreach (Programmer p in list)


           {


               Console.WriteLine(p);


           }


           Console.Read();


       }


   }


(2)总结


使用二进制序列化,必须为每一个要序列化的的类和其关联的类加上[Serializable]特性,对类中不需要序列化的成员可以使用[NonSerialized]特性。


二进制序列化对象时,能序列化类的所有成员(包括私有的),且不需要类有无参数的构造方法。


使用二进制格式序列化时,它不仅是将对象的字段数据进行持久化,也持久化每个类型的完全限定名称和定义程序集的完整名称(包括包称、版本、公钥标记、区域性),这些数据使得在进行二进制格式反序列化时亦会进行类型检查。所以反序列化时的运行环境要与序列化时的运行环境要相同,否者可能会无法反序列化成功。


3.使用SOAP方式序列化和反序列化

(1)SOAP序列化与反序列化的程序示例


   [Serializable]  //必须添加序列化特性


   public class Person


   {


       private string Name;//姓名


       private bool Sex;//性别,是否是男


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   [Serializable]  //必须添加序列化特性


   public class Programmer : Person


   {


       private string Language;//编程语言


       public Programmer(string name, bool sex, string language) : base(name, sex)


       {


           this.Language = language;


       }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           //实例化对象


           Programmer p = new Programmer("李志伟", true, "C、C#、C++、Java");


           //使用SOAP序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create, FileAccess.ReadWrite);


           SoapFormatter soapFormat = new SoapFormatter();//创建SOAP序列化器


           soapFormat.Serialize(fStream, p);//SOAP不能序列化泛型对象


           //使用SOAP反序列化对象


           fStream.Position = 0;//重置流位置


           p = null;


           p = (Programmer)soapFormat.Deserialize(fStream);


           Console.WriteLine(p);


           Console.Read();


       }


   }


(2)总结


SOAP序列化与二进制序列化的区别是:SOAP序列化不能序列化泛型类型。与二进制序列化一样在序列化时不需要向序列化器指定序列化对象的类型。而XML序列化需要向XML序列化器指定序列化对象的类型。


4.使用XML方式序列化和反序列化

示例


   public class Person


   {


       public string Name;//姓名


       public bool Sex;//性别,是否是男


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   public class Programmer : Person


   {


       public string Language;//编程语言


       public Programmer() { }//必须提供无参构造器,否则XmlSerializer将出错


       public Programmer(string name, bool sex, string language) : base(name, sex)


       {


           this.Language = language;


       }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       { //创建Programmer列表,并添加对象


           List list = new List();


           list.Add(new Programmer("李志伟", true, "C#"));


           list.Add(new Programmer("Coder2", false, "C++"));


           list.Add(new Programmer("Coder3", true, "Java"));


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create, FileAccess.ReadWrite);


           XmlSerializer xmlFormat = new XmlSerializer(


typeof(List),


new Type[] { typeof(Programmer),typeof(Person) }


);//创建XML序列化器,需要指定对象的类型


           xmlFormat.Serialize(fStream, list);


           //使用XML反序列化对象


           fStream.Position = 0;//重置流位置


           list.Clear();


           list = (List)xmlFormat.Deserialize(fStream);


           foreach (Programmer p in list)


           {


               Console.WriteLine(p);


           }


           Console.Read();


       }


   }


总结


使用XML序列化或反序列化时,需要对XML序列化器指定需要序列化对象的类型和其关联的类型。


XML序列化只能序列化对象的公有属性,并且要求对象有一个无参的构造方法,否者无法反序列化。


[Serializable]和[NonSerialized]特性对XML序列化无效!所以使用XML序列化时不需要对对象增加[Serializable]特性。


5.XML序列化对象详解

(1)说明

本节主要介绍:使用特性控制对象序列化成XML文件的格式。


(2)使用XmlElement(默认值)

类声明:


   public class Person


   {


       [XmlElement]


       public string Name;//使用[XmlElement]特性


       public bool Sex;//默认使用了[XmlElement]特性


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:



李志伟


true



(3)使用XmlAttribute

类声明:


   public class Person


   {


       [XmlElement]


       public string Name;


       [XmlAttribute]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:



李志伟



(4)使用XmlText

类声明:


   public class Person


   {


       [XmlText]


       public string Name;


       [XmlAttribute]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:


李志伟


(5)使用XmlType和XmlAttribute(重命名节点名称)

类声明:


using System.Xml.Serialization;


 [XmlType("qwe")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name { get; set; }


       [XmlAttribute("性别")]


       public bool Sex { get; set; }


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public Person() { }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   [XmlType("个人信息")]//注意:父类和子类的XmlType不能相同,相同则报错


   public class Programmer : Person


   {


       [XmlAttribute]


       public string Language { get; set; }


       public Programmer(string name, bool sex, string language)


           : base(name, sex)


       {


           this.Language = language;


       }


       public Programmer() { }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           List list = new List();


           list.Add(new Programmer("李志伟", true, "C#"));


           list.Add(new Programmer("Coder2", false, "C++"));


           list.Add(new Programmer("Coder3", true, "Java"));


           using (Stream fs = new FileStream("D:person.xml", FileMode.Truncate, FileAccess.ReadWrite))


           {


               XmlSerializer bf = new XmlSerializer(typeof(List));


               bf.Serialize(fs, list);


           }


           using (Stream fs = new FileStream("D:person.xml", FileMode.Open))


           {


               XmlSerializer bf = new XmlSerializer(typeof(List));


               List lst = bf.Deserialize(fs) as List;


               if (lst != null)


               {


                   Console.WriteLine(lst[1].ToString());


               }


           }


           Console.ReadLine();


       }


}


序列化生成的XML文件:


<个人信息xmlns:xsi="..."xmlns:xsd="..."姓名="李志伟"性别="true" />


(6)列表和数组的序列化

类声明:


[XmlType("个人信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   class Program


   {


       static void Main(string[] args)


       {


           Person p = new Person();


           p.Name = "李志伟";


           p.Sex = true;


           Person[] ps = new Person[3];


           ps[0] = p;


           ps[1] = p;


           ps[2] = p;


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(Person[]));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


序列化生成的XML文件:



 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />



注意:发现此时的XML文件的根节点名称变了。此时要重命名根节点应使用如下方式:


   [XmlType("个人信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   [XmlType("人员信息")]


   public class PersonArray : List { }


   class Program


   {


       static void Main(string[] args)


       {


           Person p = new Person();


           p.Name = "李志伟";


           p.Sex = true;


           PersonArray ps = new PersonArray();


           ps.Add(p);


           ps.Add(p);


           ps.Add(p);


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(PersonArray));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


   }


序列化生成的XML文件:


<人员信息xmlns:xsi="..."xmlns:xsd="...">


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />



(7)列表和数组的做为数据成员的序列化

类声明:


     [XmlType("信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   public class PersonArray


   {


       public List Array = new List();


       public Person Person = new Person();


   }


   class Program


   {


       static void Main(string[] args)


       {


           PersonArray ps = new PersonArray();


           ps.Person = new Person();


           ps.Person.Name = "李志伟";


           ps.Person.Sex = true;


           ps.Array.Add(ps.Person);


           ps.Array.Add(ps.Person);


           ps.Array.Add(ps.Person);


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(PersonArray));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


序列化生成的XML文件:



 


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


 


 



注意:假设这里需要为Array和Person的节点重命名,代码如下:


   [XmlType("信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   public class PersonArray


   {


       [XmlArrayItem("个人信息")]


       [XmlArray("人员信息")]


       public List Array=new List();


       public Person Person = new Person();


   }


序列化生成的XML文件:



 <人员信息>


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


 


 



注意:把“人员信息”节点去掉呢(直接出现“个人信息”节点)


   [XmlType("信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   public class PersonArray


   {


       [XmlElement("个人信息")]


       public List Array=new List();


       public Person Person = new Person();


   }


序列化生成的XML文件:



 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 



(8)类型继承与反序列化

类声明:


  public class Base { }


   [XmlType("信息A")]


   public class PersonA : Base


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public PersonA() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   [XmlType("信息B")]


   public class PersonB : Base


   {


       [XmlElement("姓名")]


       public string Name;


       [XmlElement("年龄")]


       public int Age;


       public PersonB() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   [XmlType("人员信息")]


   public class PersonArray


   {


       [XmlArrayItem(typeof(PersonA)), XmlArrayItem(typeof(PersonB))]


       public List ListPerson = new List();


   }


   class Program


   {


       static void Main(string[] args)


       {


           PersonA pa = new PersonA();


           pa.Name = "李志伟A";


           pa.Sex = true;


           PersonB pb = new PersonB();


           pb.Name = "李志伟B";


           pb.Age = 21;


           PersonArray ps = new PersonArray();


           ps.ListPerson.Add(pa);


           ps.ListPerson.Add(pa);


           ps.ListPerson.Add(pb);


           ps.ListPerson.Add(pb);


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(PersonArray));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


序列化生成的XML文件:


<人员信息xmlns:xsi="..."xmlns:xsd="...">


 


   <信息A姓名="李志伟A"性别="true" />


   <信息A姓名="李志伟A"性别="true" />


   <信息B>


     <姓名>李志伟B


     <年龄>21


 


   <信息B>


     <姓名>李志伟B


     <年龄>21


 


 



注意:同时为列表成员指定多个[XmlArrayItem(typeof(XXX))]可实现多种派生类型混在一起输出。


(9)排除不需要序列化的成员

类声明:


   public class Person


   {


       public string Name;


       [XmlIgnore]// 这个属性将不会参与序列化


       public bool Sex;


       public Person() { }


   }


序列化生成的XML文件:



李志伟



(10)强制指定成员的序列化顺序

类声明:


   public class Person


   {


       [XmlElement(Order = 2)]


       public string Name;


       [XmlElement(Order = 1)]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:



true


李志伟



(11)自定义序列化行为

类声明:


   public class Person : IXmlSerializable


   {


       public string Name;


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


       public System.Xml.Schema.XmlSchema GetSchema()


       {


           return null;


       }


       public void ReadXml(System.Xml.XmlReader reader)


       {


           Name = reader.GetAttribute("姓名");


           Sex = reader.GetAttribute("性别").Equals("男") ? true : false;


       }


       public void WriteXml(System.Xml.XmlWriter writer)


       {


           writer.WriteAttributeString("姓名", Name);


           writer.WriteAttributeString("性别", Sex ? "男" : "女");


       }


}


序列化生成的XML文件:



(12)序列化设置XML命名空间

类声明:


   [XmlRoot(Namespace = "http://msdn.microsoft.com/vsdh.xsd")]


   public class Person


   {


       public string Name;


       public bool Sex;


       public Person() { }


   }


序列化生成的XML文件:



李志伟A


true



(13)XML的使用建议

在服务端,C#代码中:


1. 建议不用使用低级别的XML API来使用XML,除非你是在设计框架或者通用类库。


2. 建议使用序列化、反序列化的方法来生成或者读取XML


3. 当需要考虑使用XML时,先不要想着XML结构,先应该定义好数据类型。


4. 列表节点不要使用[XmlElement],它会让所有子节点【升级】,显得结构混乱。


5. 如果希望序列化的XML长度小一点,可以采用[XmlAttribute],或者指定一个更短小的别名。


6. 不要在一个列表中输出不同的数据类型,这样的XML结构的可读性不好。


7. 尽量使用UTF-8编码,不要使用GB2312编码。


在客户端,JavaScript代码中,我不建议使用XML,而是建议使用JSON来代替XML,因为:


1. XML文本的长度比JSON要长,会占用更多的网络传输时间(毕竟数据保存在服务端,所以传输是免不了的)。


2. 在JavaScritp中使用XML比较麻烦(还有浏览器的兼容问题),反而各种浏览器对JSON有非常好的支持。


(14)反序列化的使用总结

如果XML是由类型序列化得到那的,那么反序列化的调用代码是很简单的,反之,如果要面对一个没有类型的XML,就需要我们先设计一个(或者一些)类型出来,这是一个逆向推导的过程,请参考以下步骤:


1. 首先要分析整个XML结构,定义与之匹配的类型,


2. 如果XML结构有嵌套层次,则需要定义多个类型与之匹配,


3. 定义具体类型(一个层级下的XML结构)时,请参考以下表格。


XML形式


处理方法


补充说明


XmlElement


定义一个属性


属性名与节点名字匹配


XmlAttribute


[XmlAttribute] 加到属性上



InnerText


[InnerText] 加到属性上


一个类型只能使用一次




节点重命名


根节点:[XmlType("testClass")]

元素节点:[XmlElement("name")]

属性节点:[XmlAttribute("id")]

列表子元素节点:[XmlArrayItem("Detail")]

列表元素自身:[XmlArray("Items")]



4.自定义序列化(仅适用于二进制与SOAP)


(1)自定义序列化的实现方式


可以通过在对象上实现 ISerializable 接口来自定义序列化过程。这一功能在反序列化后成员变量的值失效时尤其有用,但是需要为变量提供值以重建对象的完整状态。要实现ISerializable,需要实现 GetObjectData()方法以及一个特殊的构造函数,在反序列化对象时要用到此构造函数。


(2)示例程序


[Serializable]


   public class Person : ISerializable


   {


       public string Name;


       public bool Sex;


       public Person() { }


       //必须的够着方法,反序列化时调用


       protected Person(SerializationInfo info, StreamingContext context)


       {


           Name = info.GetString("姓名");


           Sex = info.GetBoolean("性别");


       }


       //序列化时调用


       public void GetObjectData(SerializationInfo info, StreamingContext context)


       {


           info.AddValue("姓名", Name + "(自定义序列化)");


           info.AddValue("性别", Sex);


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           Person p = new Person();


           p.Name = "李志伟A";


           p.Sex = true;


           //使用二进制序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           BinaryFormatter binFormat = new BinaryFormatter();//创建二进制序列化器


           binFormat.Serialize(fStream, p);//序列化对象


           //使用二进制反序列化对象


           fStream.Position = 0;//重置流位置


           p = (Person)binFormat.Deserialize(fStream);//反序列化对象


           Console.WriteLine(p);


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


注意:在序列化过程中调用 GetObjectData()时,需要填充方法调用中提供的SerializationInfo对象。只需按名称/值对的形式添加将要序列化的变量。其名称可以是任何文本。只要已序列化的数据足以在反序列化过程中还原对象,便可以自由选择添加至SerializationInfo 的成员变量。如果基对象实现了 ISerializable,则派生类应调用其基对象的 GetObjectData()方法。同样,在反序列化时也会调用含有(SerializationInfo info, StreamingContextcontext)参数的特殊的够着方法!否者将无法反序列化!!!

————————————————

版权声明:本文为CSDN博主「AI浩」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。

原文链接:https://blog.csdn.net/hhhhhhhhhhwwwwwwwwww/article/details/105362275概念:序列化是将对象状态转换为可保持或传输的形式的过程。如果一个类所创建的对象,能够被序列化,那么要求必须给这个类加上[Serializable]特性。


1.      使用二进制形式序列化对象


必须添加System.Runtime.Serialization.Formatters.Binary;命名空间


2.      使用soap形式序列化对象


3.      使用xml形式序列化对象


例:


public partial class Form1 : Form


   {

       public Form1()


       {

           InitializeComponent();


       }



       private void button1_Click(object sender, EventArgs e)


       {

           Student st = new Student() { RealName = "aa", Gender = "女" };


           FileStream fs = new FileStream(@"D:stu.txt", FileMode.OpenOrCreate);


           BinaryFormatter bf = new BinaryFormatter();


           bf.Serialize(fs, st);//把对象存放到文件流中。


           fs.Close();


       }


       ///


       /// 反序列化取出数据


       ///


       private void button2_Click(object sender, EventArgs e)


       {

           FileStream fs = new FileStream(@"D:stu.txt", FileMode.Open);


           BinaryFormatter bf = new BinaryFormatter();


           Student st = bf.Deserialize(fs) as Student;//反序列化从文件中取出数据


           if (st != null)


           {

               MessageBox.Show(st.RealName.ToString());


           }


           fs.Close();


       }


   }


   [Serializable]


   public class Student


   {

       public string RealName { get; set; }


       public String Gender { get; set; }


   }



1.对象序列化的介绍

(1).NET支持对象序列化的几种方式


二进制序列化:对象序列化之后是二进制形式的,通过BinaryFormatter类来实现的,这个类位于System.Runtime.Serialization.Formatters.Binary命名空间下。


SOAP序列化:对象序列化之后的结果符合SOAP协议,也就是可以通过SOAP 协议传输,通过System.Runtime.Serialization.Formatters.Soap命名空间下的SoapFormatter类来实现的。


XML序列化:对象序列化之后的结果是XML形式的,通过XmlSerializer 类来实现的,这个类位于System.Xml.Serialization命名空间下。XML序列化不能序列化私有数据。


(2)几种序列化的区别


二进制格式和SOAP格式可序列化一个类型的所有可序列化字段,不管它是公共字段还是私有字段。XML格式仅能序列化公共字段或拥有公共属性的私有字段,未通过属性公开的私有字段将被忽略。


使用二进制格式序列化时,它不仅是将对象的字段数据进行持久化,也持久化每个类型的完全限定名称和定义程序集的完整名称(包括包称、版本、公钥标记、区域性),这些数据使得在进行二进制格式反序列化时亦会进行类型检查。SOAP格式序列化通过使用XML命名空间来持久化原始程序集信息。而XML格式序列化不会保存完整的类型名称或程序集信息。这便利XML数据表现形式更有终端开放性。如果希望尽可能延伸持久化对象图的使用范围时,SOAP格式和XML格式是理想选择。


(3)使用特性对序列化的控制


要让一个对象支持.Net序列化服务,用户必须为每一个关联的类加上[Serializable]特性。如果类中有些成员不适合参与序列化(比如:密码字段),可以在这些域前加上[NonSerialized]特性。


2.使用二进制序列化和反序列化

(1)二进制序列化与反序列化的程序示例


   [Serializable]  //必须添加序列化特性


   public class Person


   {


       private string Name;//姓名


       private bool Sex;//性别,是否是男


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   [Serializable]  //必须添加序列化特性


   public class Programmer : Person


   {


       private string Language;//编程语言


       public Programmer(string name, bool sex, string language) : base(name, sex)


       {


           this.Language = language;


       }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           //创建Programmer列表,并添加对象


           List list = new List();


           list.Add(new Programmer("李志伟", true, "C#"));


           list.Add(new Programmer("Coder2", false, "C++"));


           list.Add(new Programmer("Coder3", true, "Java"));


           //使用二进制序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.dat";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create, FileAccess.ReadWrite);


           BinaryFormatter binFormat = new BinaryFormatter();//创建二进制序列化器


           binFormat.Serialize(fStream, list);


           //使用二进制反序列化对象


           list.Clear();//清空列表


           fStream.Position = 0;//重置流位置


           list = (List)binFormat.Deserialize(fStream);//反序列化对象


           foreach (Programmer p in list)


           {


               Console.WriteLine(p);


           }


           Console.Read();


       }


   }


(2)总结


使用二进制序列化,必须为每一个要序列化的的类和其关联的类加上[Serializable]特性,对类中不需要序列化的成员可以使用[NonSerialized]特性。


二进制序列化对象时,能序列化类的所有成员(包括私有的),且不需要类有无参数的构造方法。


使用二进制格式序列化时,它不仅是将对象的字段数据进行持久化,也持久化每个类型的完全限定名称和定义程序集的完整名称(包括包称、版本、公钥标记、区域性),这些数据使得在进行二进制格式反序列化时亦会进行类型检查。所以反序列化时的运行环境要与序列化时的运行环境要相同,否者可能会无法反序列化成功。


3.使用SOAP方式序列化和反序列化

(1)SOAP序列化与反序列化的程序示例


   [Serializable]  //必须添加序列化特性


   public class Person


   {


       private string Name;//姓名


       private bool Sex;//性别,是否是男


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   [Serializable]  //必须添加序列化特性


   public class Programmer : Person


   {


       private string Language;//编程语言


       public Programmer(string name, bool sex, string language) : base(name, sex)


       {


           this.Language = language;


       }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           //实例化对象


           Programmer p = new Programmer("李志伟", true, "C、C#、C++、Java");


           //使用SOAP序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create, FileAccess.ReadWrite);


           SoapFormatter soapFormat = new SoapFormatter();//创建SOAP序列化器


           soapFormat.Serialize(fStream, p);//SOAP不能序列化泛型对象


           //使用SOAP反序列化对象


           fStream.Position = 0;//重置流位置


           p = null;


           p = (Programmer)soapFormat.Deserialize(fStream);


           Console.WriteLine(p);


           Console.Read();


       }


   }


(2)总结


SOAP序列化与二进制序列化的区别是:SOAP序列化不能序列化泛型类型。与二进制序列化一样在序列化时不需要向序列化器指定序列化对象的类型。而XML序列化需要向XML序列化器指定序列化对象的类型。


4.使用XML方式序列化和反序列化

示例


   public class Person


   {


       public string Name;//姓名


       public bool Sex;//性别,是否是男


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   public class Programmer : Person


   {


       public string Language;//编程语言


       public Programmer() { }//必须提供无参构造器,否则XmlSerializer将出错


       public Programmer(string name, bool sex, string language) : base(name, sex)


       {


           this.Language = language;


       }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       { //创建Programmer列表,并添加对象


           List list = new List();


           list.Add(new Programmer("李志伟", true, "C#"));


           list.Add(new Programmer("Coder2", false, "C++"));


           list.Add(new Programmer("Coder3", true, "Java"));


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create, FileAccess.ReadWrite);


           XmlSerializer xmlFormat = new XmlSerializer(


typeof(List),


new Type[] { typeof(Programmer),typeof(Person) }


);//创建XML序列化器,需要指定对象的类型


           xmlFormat.Serialize(fStream, list);


           //使用XML反序列化对象


           fStream.Position = 0;//重置流位置


           list.Clear();


           list = (List)xmlFormat.Deserialize(fStream);


           foreach (Programmer p in list)


           {


               Console.WriteLine(p);


           }


           Console.Read();


       }


   }


总结


使用XML序列化或反序列化时,需要对XML序列化器指定需要序列化对象的类型和其关联的类型。


XML序列化只能序列化对象的公有属性,并且要求对象有一个无参的构造方法,否者无法反序列化。


[Serializable]和[NonSerialized]特性对XML序列化无效!所以使用XML序列化时不需要对对象增加[Serializable]特性。


5.XML序列化对象详解

(1)说明

本节主要介绍:使用特性控制对象序列化成XML文件的格式。


(2)使用XmlElement(默认值)

类声明:


   public class Person


   {


       [XmlElement]


       public string Name;//使用[XmlElement]特性


       public bool Sex;//默认使用了[XmlElement]特性


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:



李志伟


true



(3)使用XmlAttribute

类声明:


   public class Person


   {


       [XmlElement]


       public string Name;


       [XmlAttribute]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:



李志伟



(4)使用XmlText

类声明:


   public class Person


   {


       [XmlText]


       public string Name;


       [XmlAttribute]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:


李志伟


(5)使用XmlType和XmlAttribute(重命名节点名称)

类声明:


using System.Xml.Serialization;


 [XmlType("qwe")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name { get; set; }


       [XmlAttribute("性别")]


       public bool Sex { get; set; }


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public Person() { }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   [XmlType("个人信息")]//注意:父类和子类的XmlType不能相同,相同则报错


   public class Programmer : Person


   {


       [XmlAttribute]


       public string Language { get; set; }


       public Programmer(string name, bool sex, string language)


           : base(name, sex)


       {


           this.Language = language;


       }


       public Programmer() { }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           List list = new List();


           list.Add(new Programmer("李志伟", true, "C#"));


           list.Add(new Programmer("Coder2", false, "C++"));


           list.Add(new Programmer("Coder3", true, "Java"));


           using (Stream fs = new FileStream("D:person.xml", FileMode.Truncate, FileAccess.ReadWrite))


           {


               XmlSerializer bf = new XmlSerializer(typeof(List));


               bf.Serialize(fs, list);


           }


           using (Stream fs = new FileStream("D:person.xml", FileMode.Open))


           {


               XmlSerializer bf = new XmlSerializer(typeof(List));


               List lst = bf.Deserialize(fs) as List;


               if (lst != null)


               {


                   Console.WriteLine(lst[1].ToString());


               }


           }


           Console.ReadLine();


       }


}


序列化生成的XML文件:


<个人信息xmlns:xsi="..."xmlns:xsd="..."姓名="李志伟"性别="true" />


(6)列表和数组的序列化

类声明:


[XmlType("个人信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   class Program


   {


       static void Main(string[] args)


       {


           Person p = new Person();


           p.Name = "李志伟";


           p.Sex = true;


           Person[] ps = new Person[3];


           ps[0] = p;


           ps[1] = p;


           ps[2] = p;


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(Person[]));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


序列化生成的XML文件:



 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />



注意:发现此时的XML文件的根节点名称变了。此时要重命名根节点应使用如下方式:


   [XmlType("个人信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   [XmlType("人员信息")]


   public class PersonArray : List { }


   class Program


   {


       static void Main(string[] args)


       {


           Person p = new Person();


           p.Name = "李志伟";


           p.Sex = true;


           PersonArray ps = new PersonArray();


           ps.Add(p);


           ps.Add(p);


           ps.Add(p);


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(PersonArray));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


   }


序列化生成的XML文件:


<人员信息xmlns:xsi="..."xmlns:xsd="...">


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />



(7)列表和数组的做为数据成员的序列化

类声明:


     [XmlType("信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   public class PersonArray


   {


       public List Array = new List();


       public Person Person = new Person();


   }


   class Program


   {


       static void Main(string[] args)


       {


           PersonArray ps = new PersonArray();


           ps.Person = new Person();


           ps.Person.Name = "李志伟";


           ps.Person.Sex = true;


           ps.Array.Add(ps.Person);


           ps.Array.Add(ps.Person);


           ps.Array.Add(ps.Person);


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(PersonArray));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


序列化生成的XML文件:



 


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


 


 



注意:假设这里需要为Array和Person的节点重命名,代码如下:


   [XmlType("信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   public class PersonArray


   {


       [XmlArrayItem("个人信息")]


       [XmlArray("人员信息")]


       public List Array=new List();


       public Person Person = new Person();


   }


序列化生成的XML文件:



 <人员信息>


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


 


 



注意:把“人员信息”节点去掉呢(直接出现“个人信息”节点)


   [XmlType("信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   public class PersonArray


   {


       [XmlElement("个人信息")]


       public List Array=new List();


       public Person Person = new Person();


   }


序列化生成的XML文件:



 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 



(8)类型继承与反序列化

类声明:


  public class Base { }


   [XmlType("信息A")]


   public class PersonA : Base


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public PersonA() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   [XmlType("信息B")]


   public class PersonB : Base


   {


       [XmlElement("姓名")]


       public string Name;


       [XmlElement("年龄")]


       public int Age;


       public PersonB() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   [XmlType("人员信息")]


   public class PersonArray


   {


       [XmlArrayItem(typeof(PersonA)), XmlArrayItem(typeof(PersonB))]


       public List ListPerson = new List();


   }


   class Program


   {


       static void Main(string[] args)


       {


           PersonA pa = new PersonA();


           pa.Name = "李志伟A";


           pa.Sex = true;


           PersonB pb = new PersonB();


           pb.Name = "李志伟B";


           pb.Age = 21;


           PersonArray ps = new PersonArray();


           ps.ListPerson.Add(pa);


           ps.ListPerson.Add(pa);


           ps.ListPerson.Add(pb);


           ps.ListPerson.Add(pb);


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(PersonArray));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


序列化生成的XML文件:


<人员信息xmlns:xsi="..."xmlns:xsd="...">


 


   <信息A姓名="李志伟A"性别="true" />


   <信息A姓名="李志伟A"性别="true" />


   <信息B>


     <姓名>李志伟B


     <年龄>21


 


   <信息B>


     <姓名>李志伟B


     <年龄>21


 


 



注意:同时为列表成员指定多个[XmlArrayItem(typeof(XXX))]可实现多种派生类型混在一起输出。


(9)排除不需要序列化的成员

类声明:


   public class Person


   {


       public string Name;


       [XmlIgnore]// 这个属性将不会参与序列化


       public bool Sex;


       public Person() { }


   }


序列化生成的XML文件:



李志伟



(10)强制指定成员的序列化顺序

类声明:


   public class Person


   {


       [XmlElement(Order = 2)]


       public string Name;


       [XmlElement(Order = 1)]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:



true


李志伟



(11)自定义序列化行为

类声明:


   public class Person : IXmlSerializable


   {


       public string Name;


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


       public System.Xml.Schema.XmlSchema GetSchema()


       {


           return null;


       }


       public void ReadXml(System.Xml.XmlReader reader)


       {


           Name = reader.GetAttribute("姓名");


           Sex = reader.GetAttribute("性别").Equals("男") ? true : false;


       }


       public void WriteXml(System.Xml.XmlWriter writer)


       {


           writer.WriteAttributeString("姓名", Name);


           writer.WriteAttributeString("性别", Sex ? "男" : "女");


       }


}


序列化生成的XML文件:



(12)序列化设置XML命名空间

类声明:


   [XmlRoot(Namespace = "http://msdn.microsoft.com/vsdh.xsd")]


   public class Person


   {


       public string Name;


       public bool Sex;


       public Person() { }


   }


序列化生成的XML文件:



李志伟A


true



(13)XML的使用建议

在服务端,C#代码中:


1. 建议不用使用低级别的XML API来使用XML,除非你是在设计框架或者通用类库。


2. 建议使用序列化、反序列化的方法来生成或者读取XML


3. 当需要考虑使用XML时,先不要想着XML结构,先应该定义好数据类型。


4. 列表节点不要使用[XmlElement],它会让所有子节点【升级】,显得结构混乱。


5. 如果希望序列化的XML长度小一点,可以采用[XmlAttribute],或者指定一个更短小的别名。


6. 不要在一个列表中输出不同的数据类型,这样的XML结构的可读性不好。


7. 尽量使用UTF-8编码,不要使用GB2312编码。


在客户端,JavaScript代码中,我不建议使用XML,而是建议使用JSON来代替XML,因为:


1. XML文本的长度比JSON要长,会占用更多的网络传输时间(毕竟数据保存在服务端,所以传输是免不了的)。


2. 在JavaScritp中使用XML比较麻烦(还有浏览器的兼容问题),反而各种浏览器对JSON有非常好的支持。


(14)反序列化的使用总结

如果XML是由类型序列化得到那的,那么反序列化的调用代码是很简单的,反之,如果要面对一个没有类型的XML,就需要我们先设计一个(或者一些)类型出来,这是一个逆向推导的过程,请参考以下步骤:


1. 首先要分析整个XML结构,定义与之匹配的类型,


2. 如果XML结构有嵌套层次,则需要定义多个类型与之匹配,


3. 定义具体类型(一个层级下的XML结构)时,请参考以下表格。


XML形式


处理方法


补充说明


XmlElement


定义一个属性


属性名与节点名字匹配


XmlAttribute


[XmlAttribute] 加到属性上



InnerText


[InnerText] 加到属性上


一个类型只能使用一次




节点重命名


根节点:[XmlType("testClass")]

元素节点:[XmlElement("name")]

属性节点:[XmlAttribute("id")]

列表子元素节点:[XmlArrayItem("Detail")]

列表元素自身:[XmlArray("Items")]



4.自定义序列化(仅适用于二进制与SOAP)


(1)自定义序列化的实现方式


可以通过在对象上实现 ISerializable 接口来自定义序列化过程。这一功能在反序列化后成员变量的值失效时尤其有用,但是需要为变量提供值以重建对象的完整状态。要实现ISerializable,需要实现 GetObjectData()方法以及一个特殊的构造函数,在反序列化对象时要用到此构造函数。


(2)示例程序


[Serializable]


   public class Person : ISerializable


   {


       public string Name;


       public bool Sex;


       public Person() { }


       //必须的够着方法,反序列化时调用


       protected Person(SerializationInfo info, StreamingContext context)


       {


           Name = info.GetString("姓名");


           Sex = info.GetBoolean("性别");


       }


       //序列化时调用


       public void GetObjectData(SerializationInfo info, StreamingContext context)


       {


           info.AddValue("姓名", Name + "(自定义序列化)");


           info.AddValue("性别", Sex);


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           Person p = new Person();


           p.Name = "李志伟A";


           p.Sex = true;


           //使用二进制序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           BinaryFormatter binFormat = new BinaryFormatter();//创建二进制序列化器


           binFormat.Serialize(fStream, p);//序列化对象


           //使用二进制反序列化对象


           fStream.Position = 0;//重置流位置


           p = (Person)binFormat.Deserialize(fStream);//反序列化对象


           Console.WriteLine(p);


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


注意:在序列化过程中调用 GetObjectData()时,需要填充方法调用中提供的SerializationInfo对象。只需按名称/值对的形式添加将要序列化的变量。其名称可以是任何文本。只要已序列化的数据足以在反序列化过程中还原对象,便可以自由选择添加至SerializationInfo 的成员变量。如果基对象实现了 ISerializable,则派生类应调用其基对象的 GetObjectData()方法。同样,在反序列化时也会调用含有(SerializationInfo info, StreamingContextcontext)参数的特殊的够着方法!否者将无法反序列化!!!

————————————————

版权声明:本文为CSDN博主「AI浩」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。

原文链接:https://blog.csdn.net/hhhhhhhhhhwwwwwwwwww/article/details/105362275概念:序列化是将对象状态转换为可保持或传输的形式的过程。如果一个类所创建的对象,能够被序列化,那么要求必须给这个类加上[Serializable]特性。


1.      使用二进制形式序列化对象


必须添加System.Runtime.Serialization.Formatters.Binary;命名空间


2.      使用soap形式序列化对象


3.      使用xml形式序列化对象


例:


public partial class Form1 : Form


   {

       public Form1()


       {

           InitializeComponent();


       }



       private void button1_Click(object sender, EventArgs e)


       {

           Student st = new Student() { RealName = "aa", Gender = "女" };


           FileStream fs = new FileStream(@"D:stu.txt", FileMode.OpenOrCreate);


           BinaryFormatter bf = new BinaryFormatter();


           bf.Serialize(fs, st);//把对象存放到文件流中。


           fs.Close();


       }


       ///


       /// 反序列化取出数据


       ///


       private void button2_Click(object sender, EventArgs e)


       {

           FileStream fs = new FileStream(@"D:stu.txt", FileMode.Open);


           BinaryFormatter bf = new BinaryFormatter();


           Student st = bf.Deserialize(fs) as Student;//反序列化从文件中取出数据


           if (st != null)


           {

               MessageBox.Show(st.RealName.ToString());


           }


           fs.Close();


       }


   }


   [Serializable]


   public class Student


   {

       public string RealName { get; set; }


       public String Gender { get; set; }


   }



1.对象序列化的介绍

(1).NET支持对象序列化的几种方式


二进制序列化:对象序列化之后是二进制形式的,通过BinaryFormatter类来实现的,这个类位于System.Runtime.Serialization.Formatters.Binary命名空间下。


SOAP序列化:对象序列化之后的结果符合SOAP协议,也就是可以通过SOAP 协议传输,通过System.Runtime.Serialization.Formatters.Soap命名空间下的SoapFormatter类来实现的。


XML序列化:对象序列化之后的结果是XML形式的,通过XmlSerializer 类来实现的,这个类位于System.Xml.Serialization命名空间下。XML序列化不能序列化私有数据。


(2)几种序列化的区别


二进制格式和SOAP格式可序列化一个类型的所有可序列化字段,不管它是公共字段还是私有字段。XML格式仅能序列化公共字段或拥有公共属性的私有字段,未通过属性公开的私有字段将被忽略。


使用二进制格式序列化时,它不仅是将对象的字段数据进行持久化,也持久化每个类型的完全限定名称和定义程序集的完整名称(包括包称、版本、公钥标记、区域性),这些数据使得在进行二进制格式反序列化时亦会进行类型检查。SOAP格式序列化通过使用XML命名空间来持久化原始程序集信息。而XML格式序列化不会保存完整的类型名称或程序集信息。这便利XML数据表现形式更有终端开放性。如果希望尽可能延伸持久化对象图的使用范围时,SOAP格式和XML格式是理想选择。


(3)使用特性对序列化的控制


要让一个对象支持.Net序列化服务,用户必须为每一个关联的类加上[Serializable]特性。如果类中有些成员不适合参与序列化(比如:密码字段),可以在这些域前加上[NonSerialized]特性。


2.使用二进制序列化和反序列化

(1)二进制序列化与反序列化的程序示例


   [Serializable]  //必须添加序列化特性


   public class Person


   {


       private string Name;//姓名


       private bool Sex;//性别,是否是男


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   [Serializable]  //必须添加序列化特性


   public class Programmer : Person


   {


       private string Language;//编程语言


       public Programmer(string name, bool sex, string language) : base(name, sex)


       {


           this.Language = language;


       }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           //创建Programmer列表,并添加对象


           List list = new List();


           list.Add(new Programmer("李志伟", true, "C#"));


           list.Add(new Programmer("Coder2", false, "C++"));


           list.Add(new Programmer("Coder3", true, "Java"));


           //使用二进制序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.dat";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create, FileAccess.ReadWrite);


           BinaryFormatter binFormat = new BinaryFormatter();//创建二进制序列化器


           binFormat.Serialize(fStream, list);


           //使用二进制反序列化对象


           list.Clear();//清空列表


           fStream.Position = 0;//重置流位置


           list = (List)binFormat.Deserialize(fStream);//反序列化对象


           foreach (Programmer p in list)


           {


               Console.WriteLine(p);


           }


           Console.Read();


       }


   }


(2)总结


使用二进制序列化,必须为每一个要序列化的的类和其关联的类加上[Serializable]特性,对类中不需要序列化的成员可以使用[NonSerialized]特性。


二进制序列化对象时,能序列化类的所有成员(包括私有的),且不需要类有无参数的构造方法。


使用二进制格式序列化时,它不仅是将对象的字段数据进行持久化,也持久化每个类型的完全限定名称和定义程序集的完整名称(包括包称、版本、公钥标记、区域性),这些数据使得在进行二进制格式反序列化时亦会进行类型检查。所以反序列化时的运行环境要与序列化时的运行环境要相同,否者可能会无法反序列化成功。


3.使用SOAP方式序列化和反序列化

(1)SOAP序列化与反序列化的程序示例


   [Serializable]  //必须添加序列化特性


   public class Person


   {


       private string Name;//姓名


       private bool Sex;//性别,是否是男


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   [Serializable]  //必须添加序列化特性


   public class Programmer : Person


   {


       private string Language;//编程语言


       public Programmer(string name, bool sex, string language) : base(name, sex)


       {


           this.Language = language;


       }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           //实例化对象


           Programmer p = new Programmer("李志伟", true, "C、C#、C++、Java");


           //使用SOAP序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create, FileAccess.ReadWrite);


           SoapFormatter soapFormat = new SoapFormatter();//创建SOAP序列化器


           soapFormat.Serialize(fStream, p);//SOAP不能序列化泛型对象


           //使用SOAP反序列化对象


           fStream.Position = 0;//重置流位置


           p = null;


           p = (Programmer)soapFormat.Deserialize(fStream);


           Console.WriteLine(p);


           Console.Read();


       }


   }


(2)总结


SOAP序列化与二进制序列化的区别是:SOAP序列化不能序列化泛型类型。与二进制序列化一样在序列化时不需要向序列化器指定序列化对象的类型。而XML序列化需要向XML序列化器指定序列化对象的类型。


4.使用XML方式序列化和反序列化

示例


   public class Person


   {


       public string Name;//姓名


       public bool Sex;//性别,是否是男


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   public class Programmer : Person


   {


       public string Language;//编程语言


       public Programmer() { }//必须提供无参构造器,否则XmlSerializer将出错


       public Programmer(string name, bool sex, string language) : base(name, sex)


       {


           this.Language = language;


       }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       { //创建Programmer列表,并添加对象


           List list = new List();


           list.Add(new Programmer("李志伟", true, "C#"));


           list.Add(new Programmer("Coder2", false, "C++"));


           list.Add(new Programmer("Coder3", true, "Java"));


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create, FileAccess.ReadWrite);


           XmlSerializer xmlFormat = new XmlSerializer(


typeof(List),


new Type[] { typeof(Programmer),typeof(Person) }


);//创建XML序列化器,需要指定对象的类型


           xmlFormat.Serialize(fStream, list);


           //使用XML反序列化对象


           fStream.Position = 0;//重置流位置


           list.Clear();


           list = (List)xmlFormat.Deserialize(fStream);


           foreach (Programmer p in list)


           {


               Console.WriteLine(p);


           }


           Console.Read();


       }


   }


总结


使用XML序列化或反序列化时,需要对XML序列化器指定需要序列化对象的类型和其关联的类型。


XML序列化只能序列化对象的公有属性,并且要求对象有一个无参的构造方法,否者无法反序列化。


[Serializable]和[NonSerialized]特性对XML序列化无效!所以使用XML序列化时不需要对对象增加[Serializable]特性。


5.XML序列化对象详解

(1)说明

本节主要介绍:使用特性控制对象序列化成XML文件的格式。


(2)使用XmlElement(默认值)

类声明:


   public class Person


   {


       [XmlElement]


       public string Name;//使用[XmlElement]特性


       public bool Sex;//默认使用了[XmlElement]特性


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:



李志伟


true



(3)使用XmlAttribute

类声明:


   public class Person


   {


       [XmlElement]


       public string Name;


       [XmlAttribute]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:



李志伟



(4)使用XmlText

类声明:


   public class Person


   {


       [XmlText]


       public string Name;


       [XmlAttribute]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:


李志伟


(5)使用XmlType和XmlAttribute(重命名节点名称)

类声明:


using System.Xml.Serialization;


 [XmlType("qwe")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name { get; set; }


       [XmlAttribute("性别")]


       public bool Sex { get; set; }


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public Person() { }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   [XmlType("个人信息")]//注意:父类和子类的XmlType不能相同,相同则报错


   public class Programmer : Person


   {


       [XmlAttribute]


       public string Language { get; set; }


       public Programmer(string name, bool sex, string language)


           : base(name, sex)


       {


           this.Language = language;


       }


       public Programmer() { }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           List list = new List();


           list.Add(new Programmer("李志伟", true, "C#"));


           list.Add(new Programmer("Coder2", false, "C++"));


           list.Add(new Programmer("Coder3", true, "Java"));


           using (Stream fs = new FileStream("D:person.xml", FileMode.Truncate, FileAccess.ReadWrite))


           {


               XmlSerializer bf = new XmlSerializer(typeof(List));


               bf.Serialize(fs, list);


           }


           using (Stream fs = new FileStream("D:person.xml", FileMode.Open))


           {


               XmlSerializer bf = new XmlSerializer(typeof(List));


               List lst = bf.Deserialize(fs) as List;


               if (lst != null)


               {


                   Console.WriteLine(lst[1].ToString());


               }


           }


           Console.ReadLine();


       }


}


序列化生成的XML文件:


<个人信息xmlns:xsi="..."xmlns:xsd="..."姓名="李志伟"性别="true" />


(6)列表和数组的序列化

类声明:


[XmlType("个人信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   class Program


   {


       static void Main(string[] args)


       {


           Person p = new Person();


           p.Name = "李志伟";


           p.Sex = true;


           Person[] ps = new Person[3];


           ps[0] = p;


           ps[1] = p;


           ps[2] = p;


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(Person[]));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


序列化生成的XML文件:



 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />



注意:发现此时的XML文件的根节点名称变了。此时要重命名根节点应使用如下方式:


   [XmlType("个人信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   [XmlType("人员信息")]


   public class PersonArray : List { }


   class Program


   {


       static void Main(string[] args)


       {


           Person p = new Person();


           p.Name = "李志伟";


           p.Sex = true;


           PersonArray ps = new PersonArray();


           ps.Add(p);


           ps.Add(p);


           ps.Add(p);


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(PersonArray));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


   }


序列化生成的XML文件:


<人员信息xmlns:xsi="..."xmlns:xsd="...">


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />



(7)列表和数组的做为数据成员的序列化

类声明:


     [XmlType("信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   public class PersonArray


   {


       public List Array = new List();


       public Person Person = new Person();


   }


   class Program


   {


       static void Main(string[] args)


       {


           PersonArray ps = new PersonArray();


           ps.Person = new Person();


           ps.Person.Name = "李志伟";


           ps.Person.Sex = true;


           ps.Array.Add(ps.Person);


           ps.Array.Add(ps.Person);


           ps.Array.Add(ps.Person);


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(PersonArray));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


序列化生成的XML文件:



 


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


 


 



注意:假设这里需要为Array和Person的节点重命名,代码如下:


   [XmlType("信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   public class PersonArray


   {


       [XmlArrayItem("个人信息")]


       [XmlArray("人员信息")]


       public List Array=new List();


       public Person Person = new Person();


   }


序列化生成的XML文件:



 <人员信息>


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


 


 



注意:把“人员信息”节点去掉呢(直接出现“个人信息”节点)


   [XmlType("信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   public class PersonArray


   {


       [XmlElement("个人信息")]


       public List Array=new List();


       public Person Person = new Person();


   }


序列化生成的XML文件:



 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 



(8)类型继承与反序列化

类声明:


  public class Base { }


   [XmlType("信息A")]


   public class PersonA : Base


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public PersonA() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   [XmlType("信息B")]


   public class PersonB : Base


   {


       [XmlElement("姓名")]


       public string Name;


       [XmlElement("年龄")]


       public int Age;


       public PersonB() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   [XmlType("人员信息")]


   public class PersonArray


   {


       [XmlArrayItem(typeof(PersonA)), XmlArrayItem(typeof(PersonB))]


       public List ListPerson = new List();


   }


   class Program


   {


       static void Main(string[] args)


       {


           PersonA pa = new PersonA();


           pa.Name = "李志伟A";


           pa.Sex = true;


           PersonB pb = new PersonB();


           pb.Name = "李志伟B";


           pb.Age = 21;


           PersonArray ps = new PersonArray();


           ps.ListPerson.Add(pa);


           ps.ListPerson.Add(pa);


           ps.ListPerson.Add(pb);


           ps.ListPerson.Add(pb);


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(PersonArray));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


序列化生成的XML文件:


<人员信息xmlns:xsi="..."xmlns:xsd="...">


 


   <信息A姓名="李志伟A"性别="true" />


   <信息A姓名="李志伟A"性别="true" />


   <信息B>


     <姓名>李志伟B


     <年龄>21


 


   <信息B>


     <姓名>李志伟B


     <年龄>21


 


 



注意:同时为列表成员指定多个[XmlArrayItem(typeof(XXX))]可实现多种派生类型混在一起输出。


(9)排除不需要序列化的成员

类声明:


   public class Person


   {


       public string Name;


       [XmlIgnore]// 这个属性将不会参与序列化


       public bool Sex;


       public Person() { }


   }


序列化生成的XML文件:



李志伟



(10)强制指定成员的序列化顺序

类声明:


   public class Person


   {


       [XmlElement(Order = 2)]


       public string Name;


       [XmlElement(Order = 1)]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:



true


李志伟



(11)自定义序列化行为

类声明:


   public class Person : IXmlSerializable


   {


       public string Name;


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


       public System.Xml.Schema.XmlSchema GetSchema()


       {


           return null;


       }


       public void ReadXml(System.Xml.XmlReader reader)


       {


           Name = reader.GetAttribute("姓名");


           Sex = reader.GetAttribute("性别").Equals("男") ? true : false;


       }


       public void WriteXml(System.Xml.XmlWriter writer)


       {


           writer.WriteAttributeString("姓名", Name);


           writer.WriteAttributeString("性别", Sex ? "男" : "女");


       }


}


序列化生成的XML文件:



(12)序列化设置XML命名空间

类声明:


   [XmlRoot(Namespace = "http://msdn.microsoft.com/vsdh.xsd")]


   public class Person


   {


       public string Name;


       public bool Sex;


       public Person() { }


   }


序列化生成的XML文件:



李志伟A


true



(13)XML的使用建议

在服务端,C#代码中:


1. 建议不用使用低级别的XML API来使用XML,除非你是在设计框架或者通用类库。


2. 建议使用序列化、反序列化的方法来生成或者读取XML


3. 当需要考虑使用XML时,先不要想着XML结构,先应该定义好数据类型。


4. 列表节点不要使用[XmlElement],它会让所有子节点【升级】,显得结构混乱。


5. 如果希望序列化的XML长度小一点,可以采用[XmlAttribute],或者指定一个更短小的别名。


6. 不要在一个列表中输出不同的数据类型,这样的XML结构的可读性不好。


7. 尽量使用UTF-8编码,不要使用GB2312编码。


在客户端,JavaScript代码中,我不建议使用XML,而是建议使用JSON来代替XML,因为:


1. XML文本的长度比JSON要长,会占用更多的网络传输时间(毕竟数据保存在服务端,所以传输是免不了的)。


2. 在JavaScritp中使用XML比较麻烦(还有浏览器的兼容问题),反而各种浏览器对JSON有非常好的支持。


(14)反序列化的使用总结

如果XML是由类型序列化得到那的,那么反序列化的调用代码是很简单的,反之,如果要面对一个没有类型的XML,就需要我们先设计一个(或者一些)类型出来,这是一个逆向推导的过程,请参考以下步骤:


1. 首先要分析整个XML结构,定义与之匹配的类型,


2. 如果XML结构有嵌套层次,则需要定义多个类型与之匹配,


3. 定义具体类型(一个层级下的XML结构)时,请参考以下表格。


XML形式


处理方法


补充说明


XmlElement


定义一个属性


属性名与节点名字匹配


XmlAttribute


[XmlAttribute] 加到属性上



InnerText


[InnerText] 加到属性上


一个类型只能使用一次




节点重命名


根节点:[XmlType("testClass")]

元素节点:[XmlElement("name")]

属性节点:[XmlAttribute("id")]

列表子元素节点:[XmlArrayItem("Detail")]

列表元素自身:[XmlArray("Items")]



4.自定义序列化(仅适用于二进制与SOAP)


(1)自定义序列化的实现方式


可以通过在对象上实现 ISerializable 接口来自定义序列化过程。这一功能在反序列化后成员变量的值失效时尤其有用,但是需要为变量提供值以重建对象的完整状态。要实现ISerializable,需要实现 GetObjectData()方法以及一个特殊的构造函数,在反序列化对象时要用到此构造函数。


(2)示例程序


[Serializable]


   public class Person : ISerializable


   {


       public string Name;


       public bool Sex;


       public Person() { }


       //必须的够着方法,反序列化时调用


       protected Person(SerializationInfo info, StreamingContext context)


       {


           Name = info.GetString("姓名");


           Sex = info.GetBoolean("性别");


       }


       //序列化时调用


       public void GetObjectData(SerializationInfo info, StreamingContext context)


       {


           info.AddValue("姓名", Name + "(自定义序列化)");


           info.AddValue("性别", Sex);


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           Person p = new Person();


           p.Name = "李志伟A";


           p.Sex = true;


           //使用二进制序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           BinaryFormatter binFormat = new BinaryFormatter();//创建二进制序列化器


           binFormat.Serialize(fStream, p);//序列化对象


           //使用二进制反序列化对象


           fStream.Position = 0;//重置流位置


           p = (Person)binFormat.Deserialize(fStream);//反序列化对象


           Console.WriteLine(p);


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


注意:在序列化过程中调用 GetObjectData()时,需要填充方法调用中提供的SerializationInfo对象。只需按名称/值对的形式添加将要序列化的变量。其名称可以是任何文本。只要已序列化的数据足以在反序列化过程中还原对象,便可以自由选择添加至SerializationInfo 的成员变量。如果基对象实现了 ISerializable,则派生类应调用其基对象的 GetObjectData()方法。同样,在反序列化时也会调用含有(SerializationInfo info, StreamingContextcontext)参数的特殊的够着方法!否者将无法反序列化!!!

————————————————

版权声明:本文为CSDN博主「AI浩」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。

原文链接:https://blog.csdn.net/hhhhhhhhhhwwwwwwwwww/article/details/105362275概念:序列化是将对象状态转换为可保持或传输的形式的过程。如果一个类所创建的对象,能够被序列化,那么要求必须给这个类加上[Serializable]特性。


1.      使用二进制形式序列化对象


必须添加System.Runtime.Serialization.Formatters.Binary;命名空间


2.      使用soap形式序列化对象


3.      使用xml形式序列化对象


例:


public partial class Form1 : Form


   {

       public Form1()


       {

           InitializeComponent();


       }



       private void button1_Click(object sender, EventArgs e)


       {

           Student st = new Student() { RealName = "aa", Gender = "女" };


           FileStream fs = new FileStream(@"D:stu.txt", FileMode.OpenOrCreate);


           BinaryFormatter bf = new BinaryFormatter();


           bf.Serialize(fs, st);//把对象存放到文件流中。


           fs.Close();


       }


       ///


       /// 反序列化取出数据


       ///


       private void button2_Click(object sender, EventArgs e)


       {

           FileStream fs = new FileStream(@"D:stu.txt", FileMode.Open);


           BinaryFormatter bf = new BinaryFormatter();


           Student st = bf.Deserialize(fs) as Student;//反序列化从文件中取出数据


           if (st != null)


           {

               MessageBox.Show(st.RealName.ToString());


           }


           fs.Close();


       }


   }


   [Serializable]


   public class Student


   {

       public string RealName { get; set; }


       public String Gender { get; set; }


   }



1.对象序列化的介绍

(1).NET支持对象序列化的几种方式


二进制序列化:对象序列化之后是二进制形式的,通过BinaryFormatter类来实现的,这个类位于System.Runtime.Serialization.Formatters.Binary命名空间下。


SOAP序列化:对象序列化之后的结果符合SOAP协议,也就是可以通过SOAP 协议传输,通过System.Runtime.Serialization.Formatters.Soap命名空间下的SoapFormatter类来实现的。


XML序列化:对象序列化之后的结果是XML形式的,通过XmlSerializer 类来实现的,这个类位于System.Xml.Serialization命名空间下。XML序列化不能序列化私有数据。


(2)几种序列化的区别


二进制格式和SOAP格式可序列化一个类型的所有可序列化字段,不管它是公共字段还是私有字段。XML格式仅能序列化公共字段或拥有公共属性的私有字段,未通过属性公开的私有字段将被忽略。


使用二进制格式序列化时,它不仅是将对象的字段数据进行持久化,也持久化每个类型的完全限定名称和定义程序集的完整名称(包括包称、版本、公钥标记、区域性),这些数据使得在进行二进制格式反序列化时亦会进行类型检查。SOAP格式序列化通过使用XML命名空间来持久化原始程序集信息。而XML格式序列化不会保存完整的类型名称或程序集信息。这便利XML数据表现形式更有终端开放性。如果希望尽可能延伸持久化对象图的使用范围时,SOAP格式和XML格式是理想选择。


(3)使用特性对序列化的控制


要让一个对象支持.Net序列化服务,用户必须为每一个关联的类加上[Serializable]特性。如果类中有些成员不适合参与序列化(比如:密码字段),可以在这些域前加上[NonSerialized]特性。


2.使用二进制序列化和反序列化

(1)二进制序列化与反序列化的程序示例


   [Serializable]  //必须添加序列化特性


   public class Person


   {


       private string Name;//姓名


       private bool Sex;//性别,是否是男


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   [Serializable]  //必须添加序列化特性


   public class Programmer : Person


   {


       private string Language;//编程语言


       public Programmer(string name, bool sex, string language) : base(name, sex)


       {


           this.Language = language;


       }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           //创建Programmer列表,并添加对象


           List list = new List();


           list.Add(new Programmer("李志伟", true, "C#"));


           list.Add(new Programmer("Coder2", false, "C++"));


           list.Add(new Programmer("Coder3", true, "Java"));


           //使用二进制序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.dat";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create, FileAccess.ReadWrite);


           BinaryFormatter binFormat = new BinaryFormatter();//创建二进制序列化器


           binFormat.Serialize(fStream, list);


           //使用二进制反序列化对象


           list.Clear();//清空列表


           fStream.Position = 0;//重置流位置


           list = (List)binFormat.Deserialize(fStream);//反序列化对象


           foreach (Programmer p in list)


           {


               Console.WriteLine(p);


           }


           Console.Read();


       }


   }


(2)总结


使用二进制序列化,必须为每一个要序列化的的类和其关联的类加上[Serializable]特性,对类中不需要序列化的成员可以使用[NonSerialized]特性。


二进制序列化对象时,能序列化类的所有成员(包括私有的),且不需要类有无参数的构造方法。


使用二进制格式序列化时,它不仅是将对象的字段数据进行持久化,也持久化每个类型的完全限定名称和定义程序集的完整名称(包括包称、版本、公钥标记、区域性),这些数据使得在进行二进制格式反序列化时亦会进行类型检查。所以反序列化时的运行环境要与序列化时的运行环境要相同,否者可能会无法反序列化成功。


3.使用SOAP方式序列化和反序列化

(1)SOAP序列化与反序列化的程序示例


   [Serializable]  //必须添加序列化特性


   public class Person


   {


       private string Name;//姓名


       private bool Sex;//性别,是否是男


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   [Serializable]  //必须添加序列化特性


   public class Programmer : Person


   {


       private string Language;//编程语言


       public Programmer(string name, bool sex, string language) : base(name, sex)


       {


           this.Language = language;


       }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           //实例化对象


           Programmer p = new Programmer("李志伟", true, "C、C#、C++、Java");


           //使用SOAP序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create, FileAccess.ReadWrite);


           SoapFormatter soapFormat = new SoapFormatter();//创建SOAP序列化器


           soapFormat.Serialize(fStream, p);//SOAP不能序列化泛型对象


           //使用SOAP反序列化对象


           fStream.Position = 0;//重置流位置


           p = null;


           p = (Programmer)soapFormat.Deserialize(fStream);


           Console.WriteLine(p);


           Console.Read();


       }


   }


(2)总结


SOAP序列化与二进制序列化的区别是:SOAP序列化不能序列化泛型类型。与二进制序列化一样在序列化时不需要向序列化器指定序列化对象的类型。而XML序列化需要向XML序列化器指定序列化对象的类型。


4.使用XML方式序列化和反序列化

示例


   public class Person


   {


       public string Name;//姓名


       public bool Sex;//性别,是否是男


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   public class Programmer : Person


   {


       public string Language;//编程语言


       public Programmer() { }//必须提供无参构造器,否则XmlSerializer将出错


       public Programmer(string name, bool sex, string language) : base(name, sex)


       {


           this.Language = language;


       }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       { //创建Programmer列表,并添加对象


           List list = new List();


           list.Add(new Programmer("李志伟", true, "C#"));


           list.Add(new Programmer("Coder2", false, "C++"));


           list.Add(new Programmer("Coder3", true, "Java"));


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create, FileAccess.ReadWrite);


           XmlSerializer xmlFormat = new XmlSerializer(


typeof(List),


new Type[] { typeof(Programmer),typeof(Person) }


);//创建XML序列化器,需要指定对象的类型


           xmlFormat.Serialize(fStream, list);


           //使用XML反序列化对象


           fStream.Position = 0;//重置流位置


           list.Clear();


           list = (List)xmlFormat.Deserialize(fStream);


           foreach (Programmer p in list)


           {


               Console.WriteLine(p);


           }


           Console.Read();


       }


   }


总结


使用XML序列化或反序列化时,需要对XML序列化器指定需要序列化对象的类型和其关联的类型。


XML序列化只能序列化对象的公有属性,并且要求对象有一个无参的构造方法,否者无法反序列化。


[Serializable]和[NonSerialized]特性对XML序列化无效!所以使用XML序列化时不需要对对象增加[Serializable]特性。


5.XML序列化对象详解

(1)说明

本节主要介绍:使用特性控制对象序列化成XML文件的格式。


(2)使用XmlElement(默认值)

类声明:


   public class Person


   {


       [XmlElement]


       public string Name;//使用[XmlElement]特性


       public bool Sex;//默认使用了[XmlElement]特性


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:



李志伟


true



(3)使用XmlAttribute

类声明:


   public class Person


   {


       [XmlElement]


       public string Name;


       [XmlAttribute]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:



李志伟



(4)使用XmlText

类声明:


   public class Person


   {


       [XmlText]


       public string Name;


       [XmlAttribute]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:


李志伟


(5)使用XmlType和XmlAttribute(重命名节点名称)

类声明:


using System.Xml.Serialization;


 [XmlType("qwe")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name { get; set; }


       [XmlAttribute("性别")]


       public bool Sex { get; set; }


       public Person(string name, bool sex)


       {


           this.Name = name;


           this.Sex = sex;


       }


       public Person() { }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   [XmlType("个人信息")]//注意:父类和子类的XmlType不能相同,相同则报错


   public class Programmer : Person


   {


       [XmlAttribute]


       public string Language { get; set; }


       public Programmer(string name, bool sex, string language)


           : base(name, sex)


       {


           this.Language = language;


       }


       public Programmer() { }


       public override string ToString()


       {


           return base.ToString() + "\t编程语言:" + this.Language;


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           List list = new List();


           list.Add(new Programmer("李志伟", true, "C#"));


           list.Add(new Programmer("Coder2", false, "C++"));


           list.Add(new Programmer("Coder3", true, "Java"));


           using (Stream fs = new FileStream("D:person.xml", FileMode.Truncate, FileAccess.ReadWrite))


           {


               XmlSerializer bf = new XmlSerializer(typeof(List));


               bf.Serialize(fs, list);


           }


           using (Stream fs = new FileStream("D:person.xml", FileMode.Open))


           {


               XmlSerializer bf = new XmlSerializer(typeof(List));


               List lst = bf.Deserialize(fs) as List;


               if (lst != null)


               {


                   Console.WriteLine(lst[1].ToString());


               }


           }


           Console.ReadLine();


       }


}


序列化生成的XML文件:


<个人信息xmlns:xsi="..."xmlns:xsd="..."姓名="李志伟"性别="true" />


(6)列表和数组的序列化

类声明:


[XmlType("个人信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   class Program


   {


       static void Main(string[] args)


       {


           Person p = new Person();


           p.Name = "李志伟";


           p.Sex = true;


           Person[] ps = new Person[3];


           ps[0] = p;


           ps[1] = p;


           ps[2] = p;


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(Person[]));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


序列化生成的XML文件:



 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />



注意:发现此时的XML文件的根节点名称变了。此时要重命名根节点应使用如下方式:


   [XmlType("个人信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   [XmlType("人员信息")]


   public class PersonArray : List { }


   class Program


   {


       static void Main(string[] args)


       {


           Person p = new Person();


           p.Name = "李志伟";


           p.Sex = true;


           PersonArray ps = new PersonArray();


           ps.Add(p);


           ps.Add(p);


           ps.Add(p);


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(PersonArray));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


   }


序列化生成的XML文件:


<人员信息xmlns:xsi="..."xmlns:xsd="...">


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />



(7)列表和数组的做为数据成员的序列化

类声明:


     [XmlType("信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   public class PersonArray


   {


       public List Array = new List();


       public Person Person = new Person();


   }


   class Program


   {


       static void Main(string[] args)


       {


           PersonArray ps = new PersonArray();


           ps.Person = new Person();


           ps.Person.Name = "李志伟";


           ps.Person.Sex = true;


           ps.Array.Add(ps.Person);


           ps.Array.Add(ps.Person);


           ps.Array.Add(ps.Person);


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(PersonArray));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


序列化生成的XML文件:



 


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


 


 



注意:假设这里需要为Array和Person的节点重命名,代码如下:


   [XmlType("信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   public class PersonArray


   {


       [XmlArrayItem("个人信息")]


       [XmlArray("人员信息")]


       public List Array=new List();


       public Person Person = new Person();


   }


序列化生成的XML文件:



 <人员信息>


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


   <个人信息姓名="李志伟"性别="true" />


 


 



注意:把“人员信息”节点去掉呢(直接出现“个人信息”节点)


   [XmlType("信息")]


   public class Person


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   public class PersonArray


   {


       [XmlElement("个人信息")]


       public List Array=new List();


       public Person Person = new Person();


   }


序列化生成的XML文件:



 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 <个人信息姓名="李志伟"性别="true" />


 



(8)类型继承与反序列化

类声明:


  public class Base { }


   [XmlType("信息A")]


   public class PersonA : Base


   {


       [XmlAttribute("姓名")]


       public string Name;


       [XmlAttribute("性别")]


       public bool Sex;


       public PersonA() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   [XmlType("信息B")]


   public class PersonB : Base


   {


       [XmlElement("姓名")]


       public string Name;


       [XmlElement("年龄")]


       public int Age;


       public PersonB() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


   [XmlType("人员信息")]


   public class PersonArray


   {


       [XmlArrayItem(typeof(PersonA)), XmlArrayItem(typeof(PersonB))]


       public List ListPerson = new List();


   }


   class Program


   {


       static void Main(string[] args)


       {


           PersonA pa = new PersonA();


           pa.Name = "李志伟A";


           pa.Sex = true;


           PersonB pb = new PersonB();


           pb.Name = "李志伟B";


           pb.Age = 21;


           PersonArray ps = new PersonArray();


           ps.ListPerson.Add(pa);


           ps.ListPerson.Add(pa);


           ps.ListPerson.Add(pb);


           ps.ListPerson.Add(pb);


           //使用XML序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           XmlSerializer xmlFormat = new XmlSerializer(typeof(PersonArray));


           xmlFormat.Serialize(fStream, ps);//序列化对象


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


序列化生成的XML文件:


<人员信息xmlns:xsi="..."xmlns:xsd="...">


 


   <信息A姓名="李志伟A"性别="true" />


   <信息A姓名="李志伟A"性别="true" />


   <信息B>


     <姓名>李志伟B


     <年龄>21


 


   <信息B>


     <姓名>李志伟B


     <年龄>21


 


 



注意:同时为列表成员指定多个[XmlArrayItem(typeof(XXX))]可实现多种派生类型混在一起输出。


(9)排除不需要序列化的成员

类声明:


   public class Person


   {


       public string Name;


       [XmlIgnore]// 这个属性将不会参与序列化


       public bool Sex;


       public Person() { }


   }


序列化生成的XML文件:



李志伟



(10)强制指定成员的序列化顺序

类声明:


   public class Person


   {


       [XmlElement(Order = 2)]


       public string Name;


       [XmlElement(Order = 1)]


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


   }


序列化生成的XML文件:



true


李志伟



(11)自定义序列化行为

类声明:


   public class Person : IXmlSerializable


   {


       public string Name;


       public bool Sex;


       public Person() { }//必须提供无参构造器,否则XmlSerializer将出错


       public System.Xml.Schema.XmlSchema GetSchema()


       {


           return null;


       }


       public void ReadXml(System.Xml.XmlReader reader)


       {


           Name = reader.GetAttribute("姓名");


           Sex = reader.GetAttribute("性别").Equals("男") ? true : false;


       }


       public void WriteXml(System.Xml.XmlWriter writer)


       {


           writer.WriteAttributeString("姓名", Name);


           writer.WriteAttributeString("性别", Sex ? "男" : "女");


       }


}


序列化生成的XML文件:



(12)序列化设置XML命名空间

类声明:


   [XmlRoot(Namespace = "http://msdn.microsoft.com/vsdh.xsd")]


   public class Person


   {


       public string Name;


       public bool Sex;


       public Person() { }


   }


序列化生成的XML文件:



李志伟A


true



(13)XML的使用建议

在服务端,C#代码中:


1. 建议不用使用低级别的XML API来使用XML,除非你是在设计框架或者通用类库。


2. 建议使用序列化、反序列化的方法来生成或者读取XML


3. 当需要考虑使用XML时,先不要想着XML结构,先应该定义好数据类型。


4. 列表节点不要使用[XmlElement],它会让所有子节点【升级】,显得结构混乱。


5. 如果希望序列化的XML长度小一点,可以采用[XmlAttribute],或者指定一个更短小的别名。


6. 不要在一个列表中输出不同的数据类型,这样的XML结构的可读性不好。


7. 尽量使用UTF-8编码,不要使用GB2312编码。


在客户端,JavaScript代码中,我不建议使用XML,而是建议使用JSON来代替XML,因为:


1. XML文本的长度比JSON要长,会占用更多的网络传输时间(毕竟数据保存在服务端,所以传输是免不了的)。


2. 在JavaScritp中使用XML比较麻烦(还有浏览器的兼容问题),反而各种浏览器对JSON有非常好的支持。


(14)反序列化的使用总结

如果XML是由类型序列化得到那的,那么反序列化的调用代码是很简单的,反之,如果要面对一个没有类型的XML,就需要我们先设计一个(或者一些)类型出来,这是一个逆向推导的过程,请参考以下步骤:


1. 首先要分析整个XML结构,定义与之匹配的类型,


2. 如果XML结构有嵌套层次,则需要定义多个类型与之匹配,


3. 定义具体类型(一个层级下的XML结构)时,请参考以下表格。


XML形式


处理方法


补充说明


XmlElement


定义一个属性


属性名与节点名字匹配


XmlAttribute


[XmlAttribute] 加到属性上



InnerText


[InnerText] 加到属性上


一个类型只能使用一次




节点重命名


根节点:[XmlType("testClass")]

元素节点:[XmlElement("name")]

属性节点:[XmlAttribute("id")]

列表子元素节点:[XmlArrayItem("Detail")]

列表元素自身:[XmlArray("Items")]



4.自定义序列化(仅适用于二进制与SOAP)


(1)自定义序列化的实现方式


可以通过在对象上实现 ISerializable 接口来自定义序列化过程。这一功能在反序列化后成员变量的值失效时尤其有用,但是需要为变量提供值以重建对象的完整状态。要实现ISerializable,需要实现 GetObjectData()方法以及一个特殊的构造函数,在反序列化对象时要用到此构造函数。


(2)示例程序


[Serializable]


   public class Person : ISerializable


   {


       public string Name;


       public bool Sex;


       public Person() { }


       //必须的够着方法,反序列化时调用


       protected Person(SerializationInfo info, StreamingContext context)


       {


           Name = info.GetString("姓名");


           Sex = info.GetBoolean("性别");


       }


       //序列化时调用


       public void GetObjectData(SerializationInfo info, StreamingContext context)


       {


           info.AddValue("姓名", Name + "(自定义序列化)");


           info.AddValue("性别", Sex);


       }


       public override string ToString()


       {


           return "姓名:" + this.Name + "\t性别:" + (this.Sex ? "男" : "女");


       }


   }


   class Program


   {


       static void Main(string[] args)


       {


           Person p = new Person();


           p.Name = "李志伟A";


           p.Sex = true;


           //使用二进制序列化对象


           string fileName = @"D:\users\lizw\桌面\Programmers.xml";//文件名称与路径


           Stream fStream = new FileStream(fileName, FileMode.Create);


           BinaryFormatter binFormat = new BinaryFormatter();//创建二进制序列化器


           binFormat.Serialize(fStream, p);//序列化对象


           //使用二进制反序列化对象


           fStream.Position = 0;//重置流位置


           p = (Person)binFormat.Deserialize(fStream);//反序列化对象


           Console.WriteLine(p);


           fStream.Dispose();//关闭文件


           Console.WriteLine("OK!");


           Console.Read();


       }


}


注意:在序列化过程中调用 GetObjectData()时,需要填充方法调用中提供的SerializationInfo对象。只需按名称/值对的形式添加将要序列化的变量。其名称可以是任何文本。只要已序列化的数据足以在反序列化过程中还原对象,便可以自由选择添加至SerializationInfo 的成员变量。如果基对象实现了 ISerializable,则派生类应调用其基对象的 GetObjectData()方法。同样,在反序列化时也会调用含有(SerializationInfo info, StreamingContextcontext)参数的特殊的够着方法!否者将无法反序列化!!!

————————————————

版权声明:本文为CSDN博主「AI浩」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。

原文链接:https://blog.csdn.net/hhhhhhhhhhwwwwwwwwww/article/details/105362275

目录
相关文章
|
1月前
|
JSON 数据格式 索引
Python中序列化/反序列化JSON格式的数据
【11月更文挑战第4天】本文介绍了 Python 中使用 `json` 模块进行序列化和反序列化的操作。序列化是指将 Python 对象(如字典、列表)转换为 JSON 字符串,主要使用 `json.dumps` 方法。示例包括基本的字典和列表序列化,以及自定义类的序列化。反序列化则是将 JSON 字符串转换回 Python 对象,使用 `json.loads` 方法。文中还提供了具体的代码示例,展示了如何处理不同类型的 Python 对象。
|
1月前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第22天】在Java的世界里,对象序列化和反序列化是数据持久化和网络传输的关键技术。本文将带你了解如何在Java中实现对象的序列化与反序列化,并探讨其背后的原理。通过实际代码示例,我们将一步步展示如何将复杂数据结构转换为字节流,以及如何将这些字节流还原为Java对象。文章还将讨论在使用序列化时应注意的安全性问题,以确保你的应用程序既高效又安全。
|
2月前
|
存储 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第9天】在Java的世界里,对象序列化是连接数据持久化与网络通信的桥梁。本文将深入探讨Java对象序列化的机制、实践方法及反序列化过程,通过代码示例揭示其背后的原理。从基础概念到高级应用,我们将一步步揭开序列化技术的神秘面纱,让读者能够掌握这一强大工具,以应对数据存储和传输的挑战。
|
2月前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第3天】在Java编程的世界里,对象序列化与反序列化是实现数据持久化和网络传输的关键技术。本文将深入探讨Java序列化的原理、应用场景以及如何通过代码示例实现对象的序列化与反序列化过程。从基础概念到实践操作,我们将一步步揭示这一技术的魅力所在。
|
1月前
|
存储 缓存 NoSQL
一篇搞懂!Java对象序列化与反序列化的底层逻辑
本文介绍了Java中的序列化与反序列化,包括基本概念、应用场景、实现方式及注意事项。序列化是将对象转换为字节流,便于存储和传输;反序列化则是将字节流还原为对象。文中详细讲解了实现序列化的步骤,以及常见的反序列化失败原因和最佳实践。通过实例和代码示例,帮助读者更好地理解和应用这一重要技术。
36 0
|
3月前
|
JSON 安全 编译器
扩展类实例的序列化和反序列化
扩展类实例的序列化和反序列化
40 1
|
3月前
|
JSON fastjson Java
niubility!即使JavaBean没有默认无参构造器,fastjson也可以反序列化。- - - - 阿里Fastjson反序列化源码分析
本文详细分析了 Fastjson 反序列化对象的源码(版本 fastjson-1.2.60),揭示了即使 JavaBean 沲有默认无参构造器,Fastjson 仍能正常反序列化的技术内幕。文章通过案例展示了 Fastjson 在不同构造器情况下的行为,并深入探讨了 `ParserConfig#getDeserializer` 方法的核心逻辑。此外,还介绍了 ASM 字节码技术的应用及其在反序列化过程中的角色。
91 10
|
3月前
|
存储 XML JSON
用示例说明序列化和反序列化
用示例说明序列化和反序列化
22 1
|
3月前
|
存储 Java 开发者
Java编程中的对象序列化与反序列化
【9月更文挑战第20天】在本文中,我们将探索Java编程中的一个核心概念——对象序列化与反序列化。通过简单易懂的语言和直观的代码示例,你将学会如何将对象状态保存为字节流,以及如何从字节流恢复对象状态。这不仅有助于理解Java中的I/O机制,还能提升你的数据持久化能力。准备好让你的Java技能更上一层楼了吗?让我们开始吧!
|
3月前
|
XML Dubbo Java
分布式-序列化,反序列化
分布式-序列化,反序列化