剛才用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就被成功反序列化出來,以上純屬個人觀點,如果你覺得不對請指出。