朋友发来一个XML序列化的问题,从表象上看,是对子类对象进行XML序列化时,基类的一个属性丢失了。
从XmlIncludeAttribute入手
首先想到的是,序列化机制和继承本是两类不直接相关的概念,在.NET的实现里,中间有一个桥梁,XmlIncludeAttribute
//基类加入XmlInclude
[XmlInclude(typeof(b))]
public class a
{
public int aaa;
}
public class b : a
{
public int bbb;
}
...
var xs = new XmlSerializer(typeof(a));
using (var textWriter = new StringWriter())
{
xs.Serialize(textWriter, new b());
Console.WriteLine(textWriter);
}
P.S. 其实还有另一个桥梁:
var xs = new XmlSerializer(typeof(a), new Type[] { typeof(b) });
但看过问题源码之后,这个可能性无奈地被排除了。
求助Demo
因为觉得问题源码信息量有点大,一些字面量也很长,看着有些眼花,
只好打开LinqPad,仿照问题源码,逐步向Demo"加料"。
在未加入System.ComponentModel.DefaultValue这个Attribute
之前,序列化的结果都如所期。又尝试将_aaa
的默认值逻辑取消,结果也正常。算是找到了问题的入口。
盖棺定论
将Demo得出的结论发给朋友,他google到了如下资料:
PRB: XML Serialization: System.Xml.XmlSerializer Does Not Serialize Default Values
结论:现在看来,这个问题和序列化子类毫无关系,发生问题时的特定上下文,对我们解决问题的思路有些许误导。这种时候如果问题本身不是很复杂,用顺手的工具写个小Demo,相比在没有找到问题原因的前提下,便在问题源码的基础上用穷举法进行各种“尝试”,效率要可能要高很多。