C# 运行时序列化
一. 序列化与反序列的作用
为什么要有序列化呢,考虑下面这种情况,在WINFORM或者更为方便的WPF工程中,当我们进行UI设计时,可以随意的将一个控件剪切/张贴到另外一个地方。操作方便的背后是什么在发挥作用呢。控件明明是一个复杂的对象,却可以进行剪切张贴。这其中就涉及到了对象的转换。将控件这个复杂的对象转换为字节流,进行复制,然后再将字节流反转为控件对象。这样就实现了控件的随意剪切与张贴。
上述就体现了序列化的应用场合,数据传送。
定义:
序列化:将对象转换为字节流。
反序列化:将字节流转换为对象。
二.使用序列化
三.设计类可序列化,控制类的成员可序列化
1.类可序列化:
类默认是不可序列化的,但是我们在设计一个类时通常需要将其设计为可序列化,方便后续使用。使类可序列化的方法很简单,引入Attribute。在类前面加上[Serializable](是System命名空间,而不是System.Runtime.Serialization命名空间)。这样类就可序列化了。因为设置类可序列化后,默认类中的所有字段都是可序列化的。而不论其是private,protected。这样对于一些需要保护的字段就极其不利了。所以接下来就是控制类的成员可序列化。
2.控制类的成员可序列化
类型中有些字段是不希望其序列化
- 反序列化后没有意义。比如Windows内核(文件,进程,线程等)这些与进程有关,反序列化后不是同一进程,得到的信息没有意义。而序列化的目的是在于将对象转化为字节流,最终还是需要进行反序列化得到原有结果的。
- 字段可以通过简单计算得到。这样的字段没必要进行序列化。序列化反而会增加传送的数据量。
那么如何控制字段不可序列化呢
- 字段前加上[NonSerialized]即可。但是问题来了,这样我们反序列化就得不到这个字段的原有值了。像上面所述的【能够通过简单计算】得到的字段我们不希望他序列化,但设置了[NonSerialized],这样反序列后得不到想要的值。这样显然也不是我们希望看到的。解决办法就是使用[OnDeserialized]特性
- OnDeseriablized(StreamContext context)方法中重新计算标记为未序列化字段,因为有OnDeserialized特性,这样序列格式化器就会识别这个特性,对这个方法进行调用。类似的特性还有OnDeserializing,OnSerializing,OnSerialized。分别在序列化和反序列的不同时期调用。可以根据实际情况来添加对应的方法。看似已经解决了控制类的成员可序列化的问题了。但是又有一个新的问题。如果我现在在类型里添加新的字段。序列化格式化器处于性能的考虑,不会在一开始进行了字段是否可序列化的检查。而是调用时看是否能够序列化。序列化一个对象图时,只要有一个对象不能序列化【注意不是不可序列化】就会报异常。那么如果我们序列化一个类型的实例,在类型中添加新字段,然后反序列化不包含新字段的对象。那肯定会报错
- 解决2最后的问题,就可以通过OptionalFieldAttribute特性。这样格式化器就会识别这个特性,不会因为流中不包含这个字段而报错
四. 格式化器如何工作
无论是序列化,还是反序列化,还是识别特性OnSerialized等,都是格式化器在起作用。那么格式化是如何进行序列化与反序列化呢
序列化:
- 格式化器调用FormatterServices的GetSerializableMembers()方法,获取到需要序列化的字段
- 格式化器根据1取得的序列化字段调用FormatterServices的GetObjectData取得字段的对应值
- 格式化将程序集标示和类型名写入流中。
- 将1,2,取得的序列化字段和值写入流中。
反序列化:
- 读取程序集和类型名称,加载程序集
- 为对象分配内存。调用FormatterServices的Get'UninitzlizedObject()
- 初始化字段
- 初始化字段对应的值
- 将分配的对象,字段,字段对应值引用传递给FormatterServices的PopulateObjectMembers()方法
C# 运行时序列化的更多相关文章
- 《CLR Via C#》读书笔记:24.运行时序列化
一.什么是运行时序列化 序列化的作用就是将对象图(特定时间点的对象连接图)转换为字节流,这样这些对象图就可以在文件系统/网络进行传输. 二.序列化/反序列化快速入门 一般来说我们通过 FCL 提供的 ...
- 重温CLR(十八) 运行时序列化
序列化是将对象或对象图转换成字节流的过程,反序列化是将字节流转换回对象图的过程.在对象和字节流之间转换是很有用的机制. 1 应用程序的状态(对象图)可轻松保存到磁盘文件或数据库中,并在应用程序下次运行 ...
- clr via c# 运行时序列化
1,快速了解序列化----windows IO 系统,FileStream,BinaryFormatter,SoapFormatter--不支持泛型. public class SerializeRe ...
- 【C#进阶系列】24 运行时序列化
序列化是将对象或者对象图(一堆有包含关系的对象)转换成字节流的过程.而反序列化就是将字节流转为对象或对象图. 主要用于保存.传递数据,使得数据更易于加密和压缩. .NET内建了出色的序列化和反序列化支 ...
- (47)C#运行时序列化
序列化是将对象或对象图转化成字节流的过程.反序列化是将字节流转换回对象图的过程.
- ASP.NET 5运行时升级到Beta5
在Visual Studio 2015 RTM和Windows 10正式发布之前,微软把开源.NET升级到了beta5,带来了一些增强和改变.和Visual Studio 2015 RC一起安装的AS ...
- 1.2 - C#语言习惯 - 用运行时常量readonly而不是编译期常量const
C#中有两种类型的常量:编译期常量和运行时常量.二者有着截然不同的行为,使用不当将会带来性能上或正确性上的问题. 这两个问题最好都不要发生,不过若难以同时避免的话,那么一个略微慢一些但能保证正确的程序 ...
- Objective-C Runtime 运行时之一:类与对象
Objective-C语言是一门动态语言,它将很多静态语言在编译和链接时期做的事放到了运行时来处理.这种动态语言的优势在于:我们写代码时更具灵活性,如我们可以把消息转发给我们想要的对象,或者随意交换一 ...
- Objective-O Runtime 运行时初体验
Objective-C语言是一门动态语言,它将很多静态语言在编译和链接时期做的事放到了运行时来处理.这种动态语言的优势在于:我们写代码时更具灵活性,如我们可以把消息转发给我们想要的对象,或者随意交换一 ...
随机推荐
- Map,HashMap,TreeMap
一.HashMap,TreeMap差别 1.两种常规Map性能 HashMap:适用于在Map中插入.删除和定位元素. Treemap:适用于按自然顺序或自定义顺序遍历键(key). 2.总结 Has ...
- 查询优化--小表驱动大表(In,Exists区别)
Mysql 系列文章主页 =============== 本文将以真实例子来讲解小表驱动大表(In,Exists区别) 1 准备数据 1.1 创建表.函数.存储过程 参照 这篇(调用函数和存储过程批 ...
- c# datatable row
在指定索引位置插入新行 string fzmc = rs["fzmc"].ToString(); string mkmc = rs["mkmc"].ToStri ...
- SignalR Progress
Server public class ServerHub : Hub { public async Task<string> ALongTimeTask() { var p = new ...
- pip: unsupported locale setting
在终端里输入 $ export LC_ALL=C 可解决 http://stackoverflow.com/questions/36394101/pip-install-locale-error-un ...
- 微信小程序 --- 无法跳转到tab页面问题
首先检查你的跳转方法,如果是wx.navigateTo(OBJECT)或者是wx.redirectTo(OBJECT)都是无法跳转的,在微信小程序中如果需要跳转到具有tab的页面必须使用wx.swit ...
- ucsc 文件格式说明
链接附带的是ucsc各种格式文件的说明,以后遇到新类型的文件可以先来这里看看! https://www.genome.ucsc.edu/FAQ/FAQformat.html
- Minor GC、Major GC和Full GC之间的区别
在 Plumbr 从事 GC 暂停检测相关功能的工作时,我被迫用自己的方式,通过大量文章.书籍和演讲来介绍我所做的工作.在整个过程中,经常对 Minor.Major.和 Full GC 事件的使用感到 ...
- RDO Stack: No valid host was found. There are not enough hosts available.
Issue: When you launch an instance in Newton, you may find that the instance cannot be started due t ...
- java中static特殊性和final(static成员直接被访问,this不能用在static方法中,static不可访问非static)
java的static关键字 java中,static修饰的成员变量和成员方法叫静态变量和静态方法,不依赖类特定的实例,被类的所有实例共享. 静态变量或类变量 和 实例变量,区别是: 静态变量在内存中 ...