0

假设我在一个名为的模块中有这个简单的类ClassLibrary1

[DataContract]
public class Class1
{
    [DataMember]
    private DataTable _data;

    public void SetData(string key, object value)
    {
        _data = new DataTable("SomeName");
        _data.Columns.Add("Key", typeof(String));   
        _data.Columns.Add("Value", value.GetType());
        _data.Rows.Add(key, value);
    }
}

在一个名为 的单独模块中ClassLibrary2,我有以下类:

[DataContract]
public class Configuration
{
    [DataMember]
    private Class1 _obj;

    public Configuration()
    {
        _obj = new Class1();
        _obj.SetData("Key", MyEnum.Value2);
    }
}

此外,模块ClassLibrary2定义了一个名为 的枚举MyEnum,它没有标记为公共的,即它是模块内部的(如您所见,在上面的代码中使用)。

现在,在我的主模块中,我不想依赖ClassLibrary2,而是需要在运行时动态加载它。也就是说,我使用Assembly.LoadFrom,然后找到我需要使用的类型(在我的简单示例中,我只搜索名为“Configuration”的类型),并使用Activator.CreateInstance. 然后我像这样序列化实例:

var ser = new DataContractSerializer(config.GetType());
var outstream = new FileStream("c:\\test.xml", FileMode.Create);
ser.WriteObject(outstream, config);
outstream.Close();

到目前为止,一切正常。但是当我尝试像这样反序列化它时:

var instream = new FileStream("c:\\test.xml", FileMode.Open);
var conf = ser.ReadObject(instream);

我收到一个ArgumentException, 消息“列需要有效的数据类型”。我发现如果我也不会发生这种情况

  • 改为在主模块中定义 MyEnum
  • 用一个简单的替换 DataTableString key; Object value
  • 使用其他东西而不是枚举 MyEnum(例如字符串)

但是在我的实际应用中,这些选项都不是非常理想的。第一个选项将在不应相互依赖的模块之间创建强耦合。第二个选项是可能的,但工作量很大(我已经在课堂上的很多地方使用了 DataTable,直到最近才发现我必须进行序列化),第三个选项也有点混乱。

有没有其他方法可以让它工作?我还尝试通过DataContractSerializer构造函数和附加[KnownType(typeof(MyEnum))]Configuration类声明前面将 MyEnum 添加到序列化程序的已知类型列表中,但这没有帮助。

4

1 回答 1

0

底层代码只是找不到自动写在 test.xml 文件中的 ClassLibrary2 程序集全名。它需要一些帮助。

如果在反序列化之前挂钩AppDomain.AssemblyResolve 事件,则可以解决此问题,如下所示:

AppDomain.CurrentDomain.AssemblyResolve += CurrentDomainAssemblyResolve;
...
var instream = new FileStream("test.xml", FileMode.Open);
var conf = ser.ReadObject(instream);
...

private static Assembly CurrentDomainAssemblyResolve(object sender, ResolveEventArgs args)
{
    if (args.Name == "ClassLibrary2, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null") // adapt to your needs of course
        return Assembly.LoadFrom(@"mypath\ClassLibrary2.dll");

    return null; // don't know for this one
}
于 2013-06-14T09:45:44.053 回答