为什么.Net要求序列化的类必须有一个无参数的构造函数
刚才用xml序列化器,序列化一个类,结果报错说序列化的类必须带有一个无参的构造函数,好奇怪啊。为什么要有这么苛刻的条件,而且xml序列化还要求序列化的成员是public。
我以前一直觉得序列化器是一个很神奇的东西,因为它可以把一个对象保存在一个文件中,然后可以通过反序列化将文本文件还原成对象,觉得用起来很方便,而忘了思考它是怎样实现的。
先上一个例子:
[Serializable]
public class Persons:List<Person>
{
public void SaveData(string path)
{
using (FileStream fs = new FileStream(path, FileMode.Create))
{
XmlSerializer formatter = new XmlSerializer(typeof(Persons));
formatter.Serialize(fs, this);
}
} public static Persons LoadDataFromFile(string path)
{
Persons sc;
using (FileStream fs = new FileStream(path, FileMode.Open))
{
XmlSerializer formatter = new XmlSerializer(typeof(Persons));
sc = (Persons)formatter.Deserialize(fs);
}
return sc;
} }
Person类:
[Serializable]
public class Person
{
public string Name { get; set; }
public int Age { get; set; } public Person(string name)
{
Name = name;
Age = 0;
} public Person()
{
Name = string.Empty;
Age = 0;
}
}
现在想来,如果是由我来实现一个类型通用的序列化器,真是必须要有一个无参构造函数。为什么?
先回答我一个问题:
序列化器的序列化的是类的哪些成员? 答:字段,属性,一切的方法都不会参与序列化过程
有了这点就够了,我们知道类型的方法一旦经过编译就不能修改,方法的实现已经写在程序集里面了,所以我们没有必要把它序列化出来,这样只会多此一举,而且还浪费空间。我们可以将序列化的过程看成是先创建一个对象,然后再对对象里面的成员进行赋值(这就是为什么要被xml序列化的成员是publilc,public说明成员可以在类的外部进行赋值,对象一旦被创建就会为它分配空间)。
由于编译器是很笨的,他不会自动识别构造函数的参数,所以这里就需要一个无参数的构造函数来新建一个类,这样就能保证类型总能成功新建对象。(试想一下Perso对象是null你能对它的成员赋值吗?)
我认为反序列化的部分过程应该是这样
Person p=new Person();//这里必须遵守序列化的对象都带有一个无参构造函数,保证对象非null
//此处省略若干处理代码
p.Name="John";
p.Age=;
调用无参构造函数生成一个Person类,然后通过一系列的对xml的处理解析得知Person类中有两个字段属性 然后对这两个字段属性赋值
至此一个Person就被成功反序列化出来,以上纯属个人观点,如果你觉得不对请指出。
为什么.Net要求序列化的类必须有一个无参数的构造函数的更多相关文章
- 【c++】iostreeam中的类为何不可以直接定义一个无参对象呢
缘起 #include <iostream> #include <fstream> #include <sstream> using namespace std; ...
- 各种DTO类最好有 无参数的构造方法
以一下这个类为例 @Getter @Setter @ToString class Person { private String s; public Person(String s) { this.s ...
- Java 序列化工具类
import org.slf4j.Logger; import org.slf4j.LoggerFactory; import sun.misc.BASE64Decoder; import sun.m ...
- 序列化工具类({对实体Bean进行序列化操作.},{将字节数组反序列化为实体Bean.})
package com.dsj.gdbd.utils.serialize; import java.io.ByteArrayInputStream; import java.io.ByteArrayO ...
- JSON序列化必看以及序列化工具类
1.要序列化的类必须用 [DataContract] 特性标识 2.需要序列化的属性应用 [DataMember] 特性标识,没有该特性则表示不序列化该属性.类亦如此! 3.可以网络上找封装好 ...
- Python 之 Json序列化嵌套类
想要用python自已手动序列化嵌套类,就要明白两个问题: 1.Json是什么? 2.Json支持什么类型? 答案显而易见 Json就是嵌套对象 Json在python中支持列表,字典(当然也支持in ...
- HttpRunner的PB序列化工具类解决方案(python3)
背景 年初的时候团队内落地了HttpRunner3框架,简单介绍下:HttpRunner 是一款由python开发的面向 HTTP(S) 协议的开源通用测试框架,用例脚本为 YAML/JSON 格式, ...
- C# XML序列化帮助类代码
public static class XmlHelper { private static void XmlSerializeInternal(Stream stream, object o, En ...
- Qt信号槽机制的实现(面试的感悟,猜测每一个类保存的一个信号和槽的二维表,实际使用函数指针 元对象 还有类型安全的检查设定等等)
因为面试时问了我这道题,导致我想去了解信号槽到底是如何实现的,于是贴着顺序看了下源码,大致了解了整个框架.网上关于信号槽的文章也很多,但是大部分都是将如何应用的,这里我就写一下我所理解的如何实现吧, ...
随机推荐
- C#调用RAR压缩与解压
public void RARsave(string rarPatch, string rarFiles,string patch,string rarName) { ...
- stm8的IIC库的使用
一.前言 stm8是一款低功耗的MCU芯片,它具备stm32库函数和资源丰富的优势.也同时具有价格便宜,低功耗的特点.在一些项目中,能起到很好的作用.下面我介绍一下stm8的IIC硬件库函数驱动代码及 ...
- Selenium的PO模式(Page Object Model)|(Selenium Webdriver For Python)
研究Selenium + python 自动化测试有近两个月了,不能说非常熟练,起码对selenium自动化的执行有了深入的认识. 从最初无结构的代码,到类的使用,方法封装,从原始函数 ...
- 系统建设 > 医疗集团CRM系统建设步骤与分析
概述 医院客户关系管理系统(Customer Relationship Management,简称CRM)是一个完善的“以病人为中心”的管理系统,为集团/医院/总院分院/管理机构提供院前.院中.院后的 ...
- 一首诗,致亲爱的csdn
来自csdn的Rachel-Zhang姐姐 还记得--致亲爱的csdn 还记得你年轻时的摸样? 简单的文字,无瑕的脸庞. 现在的你,满脸风霜. 五粮液的广告,在我的文章中久久荡漾. 还记得当初的梦想? ...
- Android性能分析之TraceView的使用
TraceView简介 TraceView是AndroidSDK里面自带的工具,用于对Android的应用程序以及Framework层的代码进行性能分析. TraceView是图形化的工具,最终它会产 ...
- 【Java】模板方法模式
今天介绍的是模板方法模式~ 模板方法模式,由父类定制总体的框架,具体的细节由子类实现. 一般来说,模板方法中有3类方法: 抽象方法,父类声明方法待子类具体实现.比如例子的validate.save.u ...
- 添加 index_combine hint的索引
想试验一下 index_combine这个hint,于是做了如下试验. 1.创建一个具有若干index的表 SQL> create table test as select object_id, ...
- div盒子中子元素(子元素可能是盒子, 图片) 中居中的三种方法
- 数据源增量处理(Delta Proess)三大属性:Recod Mode、Delta Type、Serialization
声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...