为什么.Net要求序列化的类必须有一个无参数的构造函数

时间:2024-10-07 22:07:20

刚才用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就被成功反序列化出来,以上纯属个人观点,如果你觉得不对请指出。