.Net二进制反序列化

时间:2012-01-05 17:59:01

标签: c# wpf deserialization

我正在使用BinaryFormatter从另一个应用程序反序列化文件。它们共享基本库,并且文件中序列化的大多数类型都是已知的。

但是,也有一些类型完全未知且没有可更新的匹配。 我希望能够为这些案例丢弃反序列化。

目前我正在使用一个二进制格式化程序,该二进制格式化程序使用SurrogateSelector和SerializationBinder初始化,我用它来更新受最近版本中命名空间更改影响的类型。

var formatter = new BinaryFormatter
{
    Context = streamingContext,
    SurrogateSelector = ss,
    Binder = new ProxiedRemappingSerializationBinder(),
    FilterLevel = TypeFilterLevel.Low
};

try
{
    var deserializedObject = formatter.Deserialize(contentsStream);
    ...
}

具体而言,当.NET Framework内部尝试在已知接口的数组上设置值但具有未知值类型时,会出现问题。我得到的例外是:“System.InvalidCastException:Object不能存储在这种类型的数组中。”

那么有没有办法让反序列化过程丢弃这些类型(并且只留下空值),甚至以某种方式对SerializationBinder采取行动来防止这种异常?

提前致谢

**编辑**

除了堆栈跟踪之外,错误是典型的InvalidcastException。我在.NET Framework中调试,通过BinaryFormatter,BinaryObjectReader和ObjectManager,它在阵列上的修复阶段(当它为实例分配实际值时)失败。在InternalSetValue(& elemref,value)中打破Array.cs,第516行;是引发异常的地方。

我正在使用Binder将反序列化时间中的某些类型更改为我的应用程序知道的类型(更新版本的类型),或者更改为虚拟类。 我理解为什么会发生异常,基本上它试图在接口数组上设置不兼容的对象类型。

此类型是我动态生成的代理类型,用于保存与此实现类似的对象信息: http://holistictendencies.wordpress.com/2009/11/16/creating-proxies-in-for-round-tripping-unknown-objects-in-c-server-apps/ 因此,由于它没有实现数组接口,因此失败了。我只是希望引擎以某种方式丢弃这些案例。

1 个答案:

答案 0 :(得分:1)

我有一个类似的问题,我需要BinaryFormatter.Deserialize()一个我没有类型信息但希望从该对象的属性中获取一些值的类。

当反序列化该对象时,.NET抛出了SerializationException,抱怨未知的程序集类型。

我通过创建一个具有我需要的属性的自定义类(标记为Serializable)解决了它,然后创建了另一个继承自SerializationBinder的类:

public class MySerializationBinder : SerializationBinder
{
    public override Type BindToType(string assemblyName, string typeName)
    {
        if (assemblyName.Contains("namespace I don't have") && typeName.Contains("type info I don't have"))
                return typeof(MySubstitute);
        return Type.GetType($"{typeName}, {assemblyName}");
    }
}

[Serializable]
public class MySubstitute
{
    public string Name { get; set; }
    public string Title { get; set; }
}

注意如果它不是我要找的类型我只返回传入的内容。

然后,当您创建BinaryFormatter时,将binder属性设置为新自定义绑定器的实例:

 using (FileStream fs = new FileStream(openFileDialog1.FileName, FileMode.Open))
 {
       BinaryFormatter bf = new BinaryFormatter()   
      {
            Binder = new MySerializationBinder ()
       };                     
       mySubstitute = (MySubstitute)bf.Deserialize(fs);           
       fs.Close();
 }