.Net二进制反序列化

本文关键字:反序列化 二进制 Net | 更新日期: 2023-09-27 18:19:44

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

然而,也有一些类型是完全未知的,并且没有可更新的匹配。我希望能够为这些情况放弃反序列化。

目前,我正在使用一个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:对象不能存储在这种类型的数组中。"

那么,有没有什么方法可以让反序列化进程丢弃这些类型(只保留null),甚至以某种方式对SerializationBinder进行操作来防止这种异常?

提前感谢

**编辑**

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

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

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

.Net二进制反序列化

我遇到了一个类似的问题,我需要BinaryFormatter.Descialize()这个类,我没有它的类型信息,但想从该对象的属性中获取一些值。

到了反序列化该对象的时候,.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时,将绑定器属性设置为新自定义绑定器的实例:

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