代码改变世界

延时至调用时获取被反序列化数据的类型的实现

2010-04-07 15:34  咒语  阅读(1220)  评论(4编辑  收藏  举报

应用场景:
  设计一个任务调度系统,配置信息以XML行式保存在Tasks.config的配置文件里,该配置里不止一个任务。不同的任务,会有不同配置信息与设定。
 
解决方案1:使用XPath直接读
 优点:1. 直接;2. 灵活(配置可以千变万化)
 缺点:1. 不友好,要写一堆的读取XML数据的方法,每次有新的任务时都要重写不同的XML片断;2. 容易出错,很有可能因为写了一个错误的节点属性名称而得不到数据
 
解决方案2:使用对象序列化成XML文档
 缺点:1. 反序列化配置时必须有定义好的类型。
 优点:1. 友好,XML里的数据直接反序列化成对象的属性;2. 不容易出错,为什么呢?你肯定要先定义好类型序列化后使用,你别告诉我你是手写XML的;
 
 现在的问题是设计一个方法,解决它的缺点。就算有不同的配置我也能给你反序列化出来。那么抽出相同的部分,这部分不是我们关注的重点了。我们关注的是,如何重现不同的配置XML为实例。因为所有的对象都是继承自object的,那么,我们把扩展部分的类型就设定为object好了。经过测试发现,反序列化后的object是XmlNode[]数组。那我们要做的就是把这个XmlNode[]数组给转换为文本,然后再客户端使用的时候,将文本与定义好的类型进行反序列化。
 
代码原型:

 

代码

    [Serializable,
    XmlRoot(ElementName 
= "configuration")] 
    
public class XmlConfig
    {

        
/// <summary>
        
/// 扩展
        
/// </summary>
        [XmlElement("extend")]
        
public object Extend { getset; }

        
/// <summary>
        
/// 获取已设定的扩展类型实例
        
/// </summary>
        
/// <typeparam name="T">扩展的类型</typeparam>
        
/// <returns>扩展类实例</returns>
        public T GetExtend<T>() where T:class 
        {
            
return Serializer.XmlDeserializerFormText<T>(ExtendRawXml);
        }

        
/// <summary>
        
/// Extend扩展的Xml片断
        
/// </summary>
        
/// <returns></returns>
        protected string ExtendRawXml
        {
            
get
            {
                var nodes 
= Extend as XmlNode[];
                
if (nodes == null || nodes.Length == 0)
                    
return "<extend />";
                var w 
= new StringWriter();
                XmlWriter writer 
= new XmlTextWriter(w);
                writer.WriteStartElement(
"extend");
                
foreach (var node in nodes)
                    writer.WriteRaw(node.OuterXml);

                writer.WriteEndElement();
                writer.Close();
                
return w.ToString();
            }
        }

    }

    [Serializable,
    XmlRoot(
"extend")]
    
public class MyExtend
    {
        
public int Id { getset; }
        
public string Name { getset; }
    }