C#Xml序列化:无法(反)序列化来自另一个程序集的对象

时间:2015-11-26 13:36:54

标签: c# .net serialization xml-serialization xmlserializer

我正在尝试在C#中对一个对象进行XML序列化/反序列化。问题是该对象的类型未在与调用序列化的代码相同的程序集中声明。相反,它来自在运行时动态加载的程序集,因此在编译时不知道调用序列化的代码。

我正在尝试序列化的类型是:

//Assembly = P.dll
namespace EDFPlugin.Plugin1
{
    [Serializable]
    [XmlRoot(Namespace = "EDFPlugin.Plugin1")] 
    [XmlInclude(typeof(Options))] 
    public class Options
    {
        private string _username;
        private string _password;

        public string Username {
            get { return _username; }
            set { _username = value;}
        }

        public string Password
        {
            get { return _password; }
            set { _password = value; }
        }
    }
}

正如我之前提到的,我用来尝试序列化/反序列化此对象的代码位于一个程序集中,该程序集在编译时不知道Options类型(因为它加载{{1在运行时动态)。不过,我设法通过使用以下代码正确地序列化了类型:

P.dll

基本上,正如您所看到的,通过调用//Assembly = A.exe (doesn't know about P.dll at compile time) object value = GetOptions() //the actual type returned by this method is EDFPlugin.Plugin1.Options !! XmlSerializer valueSerializer = new XmlSerializer(value.GetType()); valueSerializer.Serialize(writer, value); 我可以解决在编译时不了解GetType()类型的问题,一切正常。

当我尝试反序列化时出现问题:

Options

由于我事先不知道有问题的类型,我基本上无法正确设置//Assembly = A.exe (doesn't know about P.dll at compile time) XmlSerializer valueSerializer = new XmlSerializer(typeof(object)); //have to use object, as I don't know the type in question... object value = valueSerializer.Deserialize(reader); //throws exception 。使用通用XmlSerializer,如上面的代码所示,会生成异常:

object

我该如何解决这个问题?

1 个答案:

答案 0 :(得分:3)

  

A.exe(编译时不知道P.dll)

因此,如果A.exe在运行时知道它,那么你应该能够动态加载EDFPlugin.Plugin1

怎么样:

XmlSerializer valueSerializer = new XmlSerializer(Type.GetType("EDFPlugin.Plugin1.Options")); 
object value = valueSerializer.Deserialize(reader);
  

但是,如果我不知道这个类型名称呢?

建议使用自定义界面将选项类别与其他类区分开,然后您就可以动态过滤并使用XmlSerializer加载它。

public interface IAmPlugin
{

}

public class Options: IAmPlugin
{
      ......
}

然后:

Assembly assembly = ... // Your Assemblie which contains plugin 

// XmlSerializer needs all possible types to Deserialize an interface 
var possibleTypes = assembly.GetTypes().Where(t => t.IsClass && t.IsAssignableFrom(typeof(IAmPlugin))).ToArray(); 

XmlSerializer serializer = new XmlSerializer(typeof(IAmPlugin), possibleTypes);
object value = valueSerializer.Deserialize(reader);

这假设您在Options类上有一个空的公共构造函数。

为什么选择界面而不是属性? 因为XmlSerializer只处理多种接口类型。