.net framework的类库中提供了三个可以用于序列化和反序列化的类,分别为BinaryFormatter、SoapFormatter和XmlSerializer。
BinaryFormatter的命名空间为System.Runtime.Serialization.Formatters.Binary,位于程序集mscorlib.dll中。
SoapFormatter的命名空间为System.Runtime.Serialization.Formatters.Soap,位于程序集System.Runtime.Serialization.Formatters.Soap.dll中。必须引用System.Runtime.Serialization.Formatters.Soap.dll。
XmlSerializer的命名空间为System.Xml.Serialization,位于程序集System.Xml.dll中。
需要说明的是,反序列化的过程(从文件-->对象的过程),不是new出来新对象,然后对其进行赋值的。反序列化的时候,不是依据类代码对各个成员进行初步赋值,然后执行构造函数的过程。在反序列化的时候,既不会为成员初赋值,也不会执行构造函数,而是直接对没有标注为[NonSerialized]的字段赋给其保存在文件中的值,而对于标注为[NonSerialized]的字段,其结果仅仅是default(FiledType),此处的FieldType是指字段的类型(注:可以利用OnSerialized方法来事后修改字段的值)。
对象序列化是将对象转换为二进制数据(字节流),反序列化是将二进制数据还原为对象。对象是稍纵即逝的,不仅程序重启、操作系统重启会造成对象的消失,就是退出函数范围等都可能造成对象的消失,序列化/反序列化就是为了保持对象的持久化
BinaryFormatter类有两个方法:
void Serialize(Stream stream, object pbj)
对象obj序列化到Stream中
object Deserialize(Stream stream)
将对象从stream中反序列化,返回值为反序列化得到的对象
对象序列化的一些注意事项
要序列化的类型必须标记为:[Serializable]
该类型的父类也必须标记为:[Serializable]
该类型中的所有成员的类型也必须标记为:[Serializable]
序列化只会对类中的字段序列化,(只能序列化一些状态信息)
类结构修改后之前序列化的内容尽量不要用了,否则可能会出错;
为什么要序列化?
保持对象的持久化,将一个复杂的对象转换流,方便我们的存储与信息交换
应用:ASP.net中进程外Session要求对象可序列化;
还有Xml序列化,应用开发中Json序列化已经代替了二进制序列化和Xml序列化等
面试题:类可序列化的条件是?
答:这个类必须标记为Serializable,如果这个类有父类,那父类也必须要编辑为Serialzable类型的,如果这类所有成员的也必须要编辑为Serializable。
具体例子如下:
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Runtime.Serialization.Formatters.Binary;
using System.Text;
using System.Threading.Tasks;
namespace BinForm
{
class Program
{
static void Main(string[] args)
{
BinaryFormatter nw = new BinaryFormatter();//初始化 BinaryFormatter 类
person per = new person();//声明一个序列化的类
per.Name = "henry";
per.Age = 100;
using (FileStream oo = File.OpenWrite(@"D:技术调研DEMODEMOTESTFile1.data")) //打开写入文件流
{
nw.Serialize(oo, per);//把对象序列化到文件流中
}
}
}
[Serializable]
public class person
{
public string Name { get; set; }
public int Age { get; set; }
}
}