我正在使用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/因此,由于它没有实现数组接口,所以它失败了。我只是想让引擎以某种方式丢弃这些案例。
我遇到了一个类似的问题,我需要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();
}