问题
使用 1.0 版本的程序集序列化的对象(具有私有MethodInfo
字段)不会使用该程序集的 1.1 版本反序列化(SerializationException
由于未找到所需的方法,将抛出 a)。
有什么改变?
我发现在 .NET 4.5 中,MemberInfo
via的序列化机制MemberInfoSerializationHolder
已经改变。在过去(直到 .NET 4.0),序列化数据是方法签名(通过简单的MethodInfo.ToString()
. 根据 .NET 源代码中的注释,他们添加了通过以下方式获得的第二个签名SerializationToString()
:
m_signature 存储有时不明确的成员的 ToString() 表示。相同方法或属性的多个重载可以相同 ToString()。m_signature2 存储对每个成员来说应该是唯一的 SerializationToString() 表示。它仅由 4.0 后的 CLR 版本编写和使用。
对于我所看到的MemberInfoSerializationHolder.GetRealObject()
使用这个(简化的)代码来解析方法(来自.NET源代码):
for (int i = 0; i < methods.Length; i++)
{
if (m_signature2 != null) // SerializationToString() signature
{
if (((RuntimeMethodInfo)methods[i]).SerializationToString().Equals(m_signature2))
{
methodInfo = methods[i];
break;
}
}
else
{
if (methods[i].ToString().Equals(m_signature))
{
methodInfo = methods[i];
break;
}
}
}
if (methodInfo == null)
throw new SerializationException(...);
在这种情况下,反序列化失败,因为m_signature2
无法找到签名,因为程序集名称包含版本信息,然后String.Equals()
将不匹配MyAssembly, Version=1.0.0.0
并MyAssembly, Version=1.1.0.0
引发异常。
问题
如果新搜索失败(至少因为与现有代码的兼容性),我希望框架将失败回到旧的搜索方法。我不明白为什么要与 a 进行此比较,毕竟在运行时解析了所有版本的程序集(并且默认情况下将加载较新的版本),我同意它无法解决那里的程序集版本,但它如果严格搜索失败,可能会删除/忽略它。String.Equals()
我知道序列化 a 很糟糕,MethodInfo
但此时此修复可能涉及太多更改(包括架构和代码),并且没有人会在旧代码中开始重构(此外,必须为旧代码和新代码保留存档的二进制兼容性版本,在两个方向)。
到目前为止我还没有尝试过,但是这个问题也适用于代表吗?是否有任何解决方案(带有属性或小的代码更改)来解决这个问题?