刚才用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要求序列化的类必须有一个无参数的构造函数的更多相关文章

  1. 【c++】iostreeam中的类为何不可以直接定义一个无参对象呢

    缘起 #include <iostream> #include <fstream> #include <sstream> using namespace std; ...

  2. 各种DTO类最好有 无参数的构造方法

    以一下这个类为例 @Getter @Setter @ToString class Person { private String s; public Person(String s) { this.s ...

  3. Java 序列化工具类

    import org.slf4j.Logger; import org.slf4j.LoggerFactory; import sun.misc.BASE64Decoder; import sun.m ...

  4. 序列化工具类({对实体Bean进行序列化操作.},{将字节数组反序列化为实体Bean.})

    package com.dsj.gdbd.utils.serialize; import java.io.ByteArrayInputStream; import java.io.ByteArrayO ...

  5. JSON序列化必看以及序列化工具类

    1.要序列化的类必须用 [DataContract] 特性标识   2.需要序列化的属性应用 [DataMember] 特性标识,没有该特性则表示不序列化该属性.类亦如此!   3.可以网络上找封装好 ...

  6. Python 之 Json序列化嵌套类

    想要用python自已手动序列化嵌套类,就要明白两个问题: 1.Json是什么? 2.Json支持什么类型? 答案显而易见 Json就是嵌套对象 Json在python中支持列表,字典(当然也支持in ...

  7. HttpRunner的PB序列化工具类解决方案(python3)

    背景 年初的时候团队内落地了HttpRunner3框架,简单介绍下:HttpRunner 是一款由python开发的面向 HTTP(S) 协议的开源通用测试框架,用例脚本为 YAML/JSON 格式, ...

  8. C# XML序列化帮助类代码

    public static class XmlHelper { private static void XmlSerializeInternal(Stream stream, object o, En ...

  9. Qt信号槽机制的实现(面试的感悟,猜测每一个类保存的一个信号和槽的二维表,实际使用函数指针 元对象 还有类型安全的检查设定等等)

    因为面试时问了我这道题,导致我想去了解信号槽到底是如何实现的,于是贴着顺序看了下源码,大致了解了整个框架.网上关于信号槽的文章也很多,但是大部分都是将如何应用的,这里我就写一下我所理解的如何实现吧, ...

随机推荐

  1. web项目开发规范整理总结

    一.类.函数.变量名命名: 1.定义类时,全部拼音的首字母必须大写:如Person,ClassDemo:(帕斯卡命名法):也可以用带下斜杆的匈牙利命名法进行命名,如    head_navigatio ...

  2. HTML 5 应用程序缓存(上)

    什么是应用程序缓存(Application Cache)?HTML5 引入了应用程序缓存,这意味着 web 应用可进行缓存,并可在没有因特网连接时进行访问. 应用程序缓存为应用带来三个优势: 离线浏览 ...

  3. [转]DataURL与File,Blob,canvas对象之间的互相转换的Javascript

    来源 http://blog.csdn.net/cuixiping/article/details/45932793 canvas转换为dataURL (从canvas获取dataURL) var d ...

  4. HDU2586How far away ?

    http://acm.hdu.edu.cn/showproblem.php?pid=2586 How far away ? Time Limit: 2000/1000 MS (Java/Others) ...

  5. SQL中常用模糊查询的四种匹配模式&&正则表达式

    执行数据库查询时,有完整查询和模糊查询之分.一般模糊语句如下:SELECT 字段 FROM 表 WHERE 某字段 Like 条件 其中关于条件,SQL提供了四种匹配模式:1.%:表示任意0个或多个字 ...

  6. mac上eclipse用gdb调试(转)

    mac上eclipse用gdb调试 With its new OS release, Apple has discontinued the use of GDB in OS X. Since 2005 ...

  7. DPI

    [iOS]查找数组NSArray中是否包含指定的元素 http://blog.csdn.net/zyq527758142/article/details/51278172 Dpi(每平方英寸像素数目) ...

  8. BASH比较大小

  9. inline-boock的默认间距

    1 2 3 4 div{width:900px;} div li{ display:inline-block; width:300px;} <ul>     <li></ ...

  10. phprpc的简单使用

    PHPRPC 是一个轻型的.安全的.跨网际的.跨语言的.跨平台的.跨环境的.跨域的.支持复杂对象传输的.支持引用参数传递的.支持内容输出重定向的.支持分级错误处理的.支持会话的.面向服务的高性能远程过 ...