以前一直沒搞懂為什么C#在做對象序列化時(Json序列化,XML序列化等)有時候會出現循環引用的問題,下面寫了個例子,類People有一個屬性引用了類Child,而類Child也有一個屬性引用了類People,並且兩個屬性的get訪問器中都會new一個彼此類型的對象,這樣在訪問People類的Child屬性的時候就會new一個Child對象,在訪問Child類的People屬性的時候又會new一個People對象。所以C#序列化方法在序列化People類的Child屬性時又會去序列化Child屬性返回的Child對象,而在序列化Child類的People屬性時又會去序列化People屬返回的People對象。那么C#序列化方法在檢測People類和Child類的屬性的時候就會不停地去構造(不停地new)且不停地序列化People對象和Child對象,形成一個死循環:People->Child->People->Child->People....,所以就會拋出循環引用的異常,注意下面XML序列化的時候只有屬性聲明了set訪問器才會出現循環引用的異常,說明C#自帶的XmlSerializer比JavaScriptSerializer要更先進一些。。。
類Child:
1 using System.Xml; 2 using System.Xml.Serialization; 3 4 namespace SE 5 { 6 [XmlRoot] 7 public class Child 8 { 9 public Child() 10 { 11 } 12 13 [XmlElement] 14 public string Name { get; set; } 15 16 [XmlElement] 17 public int Age { get; set; } 18 19 20 [XmlElement] 21 public People People 22 { 23 //注意屬性聲明了set訪問器XmlSerializer在序列化時才會出現循環引用異常 24 /*set 25 { 26 27 }*/ 28 get 29 { 30 return new People(); 31 } 32 } 33 } 34 }
類People:
1 using System.Xml; 2 using System.Xml.Serialization; 3 4 namespace SE 5 { 6 [XmlRoot] 7 public class People 8 { 9 public People() 10 { 11 } 12 13 [XmlElement] 14 public string Name { get; set; } 15 16 [XmlElement] 17 public int Age { get; set; } 18 19 20 [XmlElement] 21 public Child Child 22 { 23 //注意屬性聲明了set訪問器XmlSerializer在序列化時才會出現循環引用異常 24 /*set 25 { 26 27 }*/ 28 get 29 { 30 return new Child(); 31 } 32 } 33 } 34 }
序列化調用代碼:
1 using System; 2 using System.Web.Script.Serialization; 3 using System.IO; 4 using System.Xml.Serialization; 5 6 namespace SE 7 { 8 class Program 9 { 10 static void Main(string[] args) 11 { 12 People P = new People() { Name = "James", Age = 15}; 13 XmlSerializer xmls = new XmlSerializer(P.GetType()); 14 string content; 15 using (MemoryStream ms = new MemoryStream()) 16 { 17 xmls.Serialize(ms, P);//如果People類和Child類的屬性有set訪問器則拋出循環引用異常 18 ms.Position = 0; 19 using (StreamReader s = new StreamReader(ms)) 20 { 21 content = s.ReadToEnd(); 22 } 23 } 24 25 Console.ReadLine(); 26 27 JavaScriptSerializer js = new JavaScriptSerializer(); 28 content=js.Serialize(P);//拋出循環引用異常 29 Console.ReadLine(); 30 } 31 } 32 }