我正在使用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/ 因此,由于它没有实现数组接口,因此失败了。我只是希望引擎以某种方式丢弃这些案例。
答案 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();
}