.Net组件程序设计之序列化

自动序列化

本篇给大家讲解一下在.NET中的序列化,它的用处非常之多,特别是对于某种环境下保存某种状态是很好的方法,接下来就来看一下吧。

Serializable属性

     [Serializable]
public class SerializableCase { public SerializableCase() { } private string _State; public string State { get { return _State; } set { _State = value; } } }

在上面的示例类型上加上Serializable属性这样将示例类型标记为可序列化类型.

格式化序列化器

二进制格式器

     public class MySerializableCase
{
public static void BinaryFormatterSerialize()
{
IFormatter formatter = new BinaryFormatter();
Stream stream = new FileStream("jin.glory", FileMode.Create, FileAccess.ReadWrite);
using (stream)
{
SerializableCase serCase = new SerializableCase();
serCase.State = "Test";
formatter.Serialize(stream, serCase);
}
} public static void BinaryFormatterDesSerialize()
{
Stream stream = new FileStream("jin.glory", FileMode.Open, FileAccess.Read);
IFormatter formatter = new BinaryFormatter();
using (stream)
{
SerializableCase serCase = formatter.Deserialize(stream) as SerializableCase;
return serCase.State;
}
}
}

BinaryFormattSerialize()方法里只是实例化SerializableCase类型,然后对State属性赋值,代表一个状态。调用 MySerializableCase.BinaryFormattSerialize()后.NET把序列化好的文件流保存到了jin.glory文件中.

图1

文件的名称和后缀格式都是自己随便定义的。然后再调用反序列化,获取到之前序列化的对象状态。

   string state = MySerializableCase.BinaryFormattDesSerialize();
Console.WriteLine(state);

图2

SOAP格式器

SoapFormatter是在命名空间System.Runtime.Serialization.Formatters.Soap下的(在System.Runtime.Serialization.Formatters.Soap.dll中)

 public class MySerializableCase
{
public static void SoapFormatterSerialize()
{
IFormatter formatter = new SoapFormatter();
Stream stream = new FileStream("Soap.xml", FileMode.Create, FileAccess.ReadWrite);
using (stream)
{
SerializableCase serCase = new SerializableCase();
serCase.State = "Test";
formatter.Serialize(stream, serCase);
}
} public static string SoapFormatterDesSerialize()
{
Stream stream = new FileStream("Soap.xml", FileMode.Open, FileAccess.Read);
IFormatter formatter = new SoapFormatter();
using (stream)
{
SerializableCase serCase = formatter.Deserialize(stream) as SerializableCase;
return serCase.State;
}
}
}

和上面的二进制格式化器使用的方式近乎相同,唯一不同的是,使用Soap格式化器生成的序列化文件耗时更长,占用空间也比较大,但是它是遵循着SOAP协议的,这在跨平台操作数据间是很有用的,平台只需要解析重建就能把对象重新的生成出来。

不可序列化成员

     [Serializable]
public class SerializableCase
{
public SerializableCase() { } private string _State; public string State
{
get { return _State; }
set { _State = value; }
} [NonSerialized]
private DonotSerializable _DonotSerializable; public DonotSerializable DonotSerializable
{
get { return _DonotSerializable; }
set { _DonotSerializable = value; }
}
}
public class DonotSerializable
{
public DonotSerializable() { } public string DonotSerializableData
{
get;
set;
}
}

修改了一下第一段的示例代码,增加了个属性,并且设置其字段为NonSerialized,这样在.NET序列化这个类的实例的时候,检测到了[NonSerialized]的时候就会跳过它,因为有的对象确实是不适合序列化来进行持久化的,不过这样的做的也会有个问题,就是对象的状态丢失,就是不可序列化的部分会丢失。看一下下面的代码:

     public class MySerializableCase
{
public static void BinaryFormatterSerialize()
{
IFormatter formatter = new BinaryFormatter();
Stream stream = new FileStream("jin.glory", FileMode.Create, FileAccess.ReadWrite);
using (stream)
{
SerializableCase serCase = new SerializableCase();
serCase.State = "Test";
serCase.DonotSerializable = new DonotSerializable();
serCase.DonotSerializable.DonotSerializableData = "DonotSerializableData";
formatter.Serialize(stream, serCase);
}
} public static string BinaryFormatterDesSerialize()
{
Stream stream = new FileStream("jin.glory", FileMode.Open, FileAccess.Read);
IFormatter formatter = new BinaryFormatter();
using (stream)
{
SerializableCase serCase = formatter.Deserialize(stream) as SerializableCase;
return serCase.State+"_"+serCase.DonotSerializable.DonotSerializableData;
}
}
}

修改了上面的二进制格式器的代码,调用一下测试代码我们一起来看下结果:

 MySerializableCase.BinaryFormatterSerialize();
string state = MySerializableCase.BinaryFormatterDesSerialize();
Console.WriteLine(state);

图3

在反序列化的时候,对象SerializableCase的DonotSerializable属性丢失了,所以才会报错。
对于这样的情况,.NET提供了IDeserializationCallback接口,它里面只有一个函数void OnDeserialization(object sender),只要实现了IDeserializationCallback,并且在OnDeserialization函数里实现具体的对不可序列化对象的初始化。

     [Serializable]
public class SerializableCase:IDeserializationCallback
{
public SerializableCase() { } private string _State; public string State
{
get { return _State; }
set { _State = value; }
} [NonSerialized]
private DonotSerializable _DonotSerializable; public DonotSerializable DonotSerializable
{
get { return _DonotSerializable; }
set { _DonotSerializable = value; }
} public void OnDeserialization(object sender)
{
_DonotSerializable = new DonotSerializable();
_DonotSerializable.DonotSerializableData = "DonotSerializableData->Test";
}
}

按照上面的调用方式,来看一下结果:

图4

这样是在反序列化的时候,如果检测到了实例类型实现了IDeserializationCallback接口,是在反序列化完成的时候会执行实现了IDeserializationCallback的OnDeserialization()方法,这样可以对一些不可序列化的属性状态在这个方法里来实现。

序列化事件

.NET2.0
引进了对序列化事件的支持,当序列化和反序列化的时候,.NET在你的类上调用指定的方法,.NET中定义了四个序列化和反序列化事件。
serializing事件是在序列化发生之前被触发,
serialized 事件是在序列化之后被触发,
deserializing事件是在反序列化之前被触发,
deserialized事件是在反序列化之后被触发。

引用先前的示例代码SerializableCase类的初始代码:

    [Serializable]
public class SerializableCase //:IDeserializationCallback
{
public SerializableCase() { }
private string _State;
public string State
{
get { return _State; }
set { _State = value; }
}
}

添加了事件后的示例代码是这样的:

     [Serializable]
public class SerializableCase
{
public SerializableCase() { }
private string _State;
public string State
{
get { return _State; }
set { _State = value; }
} [OnSerializing]
private void OnSerializing(StreamingContext context)
{
_State = "此时的状态是:序列化之前";
Console.WriteLine(State);
} [OnSerialized]
private void OnSerialized(StreamingContext context)
{
_State = "此时的状态是:序列化之后";
Console.WriteLine(State);
} [OnDeserializing]
private void OnDeserializing(StreamingContext context)
{
_State = "此时的状态是:反序列化之前";
Console.WriteLine(State);
} [OnDeserialized]
private void OnDeserialized(StreamingContext context)
{
_State = "此时的状态是:反序列化之后";
Console.WriteLine(State);
}
}

使用之前定义好的MySerializableCase类型中的静态方法,稍作修改来演示一下,

         public static void SoapFormatterSerialize()
{
IFormatter formatter = new SoapFormatter();
Stream stream = new FileStream("Soap.xml", FileMode.Create, FileAccess.ReadWrite);
using (stream)
{
SerializableCase serCase = new SerializableCase();
formatter.Serialize(stream, serCase);
}
}
public static string SoapFormatterDesSerialize()
{
Stream stream = new FileStream("Soap.xml", FileMode.Open, FileAccess.Read);
IFormatter formatter = new SoapFormatter();
using (stream)
{
SerializableCase serCase = formatter.Deserialize(stream) as SerializableCase;
return serCase.State;
} }

测试代码:

 MySerializableCase.SoapFormatterSerialize();
MySerializableCase.SoapFormatterDesSerialize();
Console.ReadLine();

图5

从结果中就很明显的显示出来了,这里要说几句题外话,细心的朋友可能发现了,在SerializableCase类型中的四个事件函数签名都是相同的,这是因为在.NET中为这个这个序列化和反序列化事件定义的委托就是这个签名,在这个类型实例序列化和反序列化的时候会检测到.NET会反射实例内部所有的函数 检测是否有附加了序列化事件,如果判断了有则会继续检查这个函数的签名,如果函数签名也匹配了,就会把这个函数挂上委托。

可以这样指定一个函数为事件指定调用函数:

 [OnSerializing]
[OnSerialized]
[OnDeserializing]
[OnDeserialized]
private void OnGenericSerializEventMethod(StreamingContext context)
{
……
}

本篇幅序列化内容结束。

作者:金源

出处:http://www.cnblogs.com/jin-yuan/

本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面

.Net组件程序设计之序列化的更多相关文章

  1. .Net组件程序设计之远程调用(一)

    .Net组件程序设计之远程调用(一) 1应用程序域 我们知道我们写的C#代码是在操作系统逻辑体系结构中最上层的,然而操作系统本身是不会认识C#代码的,它只认识机器代码.那我们写的程序经过编译后是编译成 ...

  2. .Net组件程序设计之远程调用(二)

    .Net组件程序设计之远程调用(二) 激活模式 引用封送对象激活类型两种, 一种是客户端激活类型,一种是服务器端激活. 客户端激活对象 客户端激活方式:当客户端创建一个远程对象时,客户端得到的是一个新 ...

  3. .NET组件程序设计之线程、并发管理(二)

    .Net组件程序设计之线程.并发管理(二) 2.同步线程 手动同步 监视器 互斥 可等待事件 同步线程 所有的.NET组件都支持在多线程的环境中运行,可以被多个线程并发访问,如果没有线程同步,这样的后 ...

  4. .Net组件程序设计之线程、并发管理(一)

    .Net组件程序设计之线程.并发管理(一) 1.线程 线程 线程的创建 线程的阻塞 线程挂起 线程睡眠 加入线程 线程中止 现在几乎所有的应用程序都是多线程的,给用户看来就是一个应用程序界面(应用程序 ...

  5. .Net组件程序设计之上下文

    .Net组件程序设计之上下文 在后续篇幅的远程调用的文章里有说到应用程序域,那是大粒度的控制程序集的逻辑存在,那么想对对象的控制又由谁来做主呢?没错了,就是上下文.CLR把应用程序域更细化了,在应用程 ...

  6. .Net组件程序设计之对象生命周期

    .Net组件程序设计之对象生命周期 .NET 垃圾回收 IDisposable() Using语句 .NET 垃圾回收 是CLR管理着垃圾回收器,垃圾回收器监控着托管堆,而我们使用的对象以及系统启动是 ...

  7. .Net组件程序设计之异步调用

    .Net组件程序设计之异步调用 说到异步调用,在脑海中首先想到就是BeginInvoke(),在一些常用对象中我们也会常常见到Invoke()和BeginInvoke(), 要想让自己的组件可以被客户 ...

  8. .Net组件程序设计

    .Net组件程序设计之上下文 在后续篇幅的远程调用的文章里有说到应用程序域,那是大粒度的控制程序集的逻辑存在,那么想对对象的控制又由谁来做主呢?没错了,就是上下文.CLR把应用程序域更细化了,在应用程 ...

  9. Json.Net组件指定/忽略序列化字段属性技巧知识点

    我们在用Json.Net序列化组件序列化类的时候,经常有这样的一个需求:指定被序列化类中的某些字段属性是要忽略的,或者是指定字段属性序列化 比如下面这个类: public class Bar { pu ...

随机推荐

  1. oracle函数简析

    (一).数值型函数(Number Functions) 数值型函数输入数字型参数并返回数值型的值.多数该类函数的返回值支持38位小数点,诸如:COS, COSH, EXP, LN, LOG, SIN, ...

  2. html学习第三天—— 第12章——css布局模型

    清楚了CSS 盒模型的基本概念. 盒模型类型, 我们就可以深入探讨网页布局的基本模型了.布局模型与盒模型一样都是 CSS 最基本. 最核心的概念. 但布局模型是建立在盒模型基础之上,又不同于我们常说的 ...

  3. 搭建selenium grid简单配置

    1.使用selenium提供的服务端独立jar包 :服务端.客户端都是运行于java7环境. 2.启动hub: hub配置文件如下: Java -jar selenium-server-standal ...

  4. Codeforces CF#628 Education 8 F. Bear and Fair Set

    F. Bear and Fair Set time limit per test 2 seconds memory limit per test 256 megabytes input standar ...

  5. 【转】Oracle索引HINT的使用

    转自:Oracle索引HINT的使用       存储在数据库中数据的分布情况开发人员或管理员比Oracle优化器更加的清楚,在优化器不能作出最有查询路径选择的情况下,使用HINT(提示)人为的固定查 ...

  6. [ACM训练] 算法初级 之 数据结构 之 栈stack+队列queue (基础+进阶+POJ 1338+2442+1442)

    再次面对像栈和队列这样的相当基础的数据结构的学习,应该从多个方面,多维度去学习. 首先,这两个数据结构都是比较常用的,在标准库中都有对应的结构能够直接使用,所以第一个阶段应该是先学习直接来使用,下一个 ...

  7. SVG文件:从Illustrator导文件到Web

    可缩放矢量图形(SVG)是早在1998年就已经有的一种矢量图像格式.它总是和Web一起发展,但是直到现在才开始赶上Web发展的步伐.如今我们已经不能否认SVG和Web的相关性,所以让我们来学习一下从I ...

  8. jquery+bootstrap实现tab切换, 每次切换时都请求数据, 点击提交分别向不同的地址提交数据

    今天一个朋友叫帮做一个tab切换, 每一个tab内容区域都是从后台取出的数据, 这些数据要用表格的形式显示处理, 并且表格的内容区域可以修改, 如下所示: 例子查看请演示查看. 截图如图所示: 实现步 ...

  9. Jsonp原理就是这么简单

    原理就是:包裹数据的js数据文件,自动执行,找到目标函数,通过传参,把数据注入进去. 当你打开本篇博文,证明你已经大体知道了Jsonp的作用了. 但如果需要我介绍一下,我也可以简单介绍: 简单说,就是 ...

  10. oracle连接问题【转载】

    SQL的四种连接-左外连接.右外连接.内连接.全连接   今天在看一个遗留系统的数据表的时候发现平时查找的视图是FULL OUT JOIN的,导致平时的数据记录要进行一些限制性处理,其实也可以设置视图 ...