刚才用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=10;



调用无参构造函数生成一个Person类,然后通过一系列的对xml的处理解析得知Person类中有两个字段属性 然后对这两个字段属性赋值

至此一个Person就被成功反序列化出来,以上纯属个人观点,如果你觉得不对请指出。