在 C# 中的二进制反序列化期间更改类型

Changing types during binary deserialization in C#

我们公司的一个解决方案使用第 3 方服务。通信是通过 XML 消息传递完成的。在我们这边,我们根据他们提供给我们的 XML 模式生成要使用的 类,并且在某些时候我们将其中一些类型序列化为数据库中的二进制 blob 以供以后使用。

问题出在第 3 方公司将其中一个字段从布尔值类型更改为整数类型。现在,当我们尝试反序列化已经存在的数据时,我们可以预见到类型转换异常(无法从布尔值转换为整数)。

我的问题是 - 我们如何使用旧的布尔类型反序列化数据库中的现有数据以将其转换为新的整数类型?

我已经尝试了很多事情 - 其中包括反射和实现 ISerializable,但到目前为止还没有成功。理想的解决方案是实现 ISerializable,但在尝试反序列化现有数据时出现 "Member not found" 错误,因为它已经仅使用 Serializable 属性进行了序列化。

欢迎提出任何建议!

编辑:添加一些代码以清楚地说明我的问题。

namespace ClassLibrary
{
    [Serializable]
    public class Foo //: ISerializable
    {
        public bool Bar { get; set; }

        public Foo() { }

        //[OnDeserializing()]
        //internal void OnDeserializingMethod(StreamingContext context)
        //{
        //    Bar = 10;
        //}

        //public Foo(SerializationInfo info, StreamingContext context)
        //{
        //    Bar = (int)info.GetValue("Bar", typeof(int));
        //}

        //public void GetObjectData(SerializationInfo info, StreamingContext context)
        //{
        //    info.AddValue("Bar", Bar);
        //}
    }
}

namespace ConsoleApplication2
{
    static class Program
    {
        static void Main(string[] args)
        {
            Foo foo;

            // Run #1, where Foo.Bar is a boolean

            foo = new Foo();
            foo.Bar = true;
            SerializeObject(foo);
            byte[] data = File.ReadAllBytes(@".\Test.bin");
            foo = DeserializeObject(data) as Foo;

            // Now change Foo.Bar to an integer type, comment the lines above, and uncomment the two lines below
            //byte[] newData = File.ReadAllBytes(@".\Test.bin");
            //foo = DeserializeObject(newData) as Foo;

            Console.WriteLine(foo.Bar);
            Console.ReadLine();
        }

        private static Object DeserializeObject(byte[] buff)
        {
            if (buff == null) return null;
            BinaryFormatter formatter = new BinaryFormatter();
            formatter.Binder = new CustomSerializationBinder();
            MemoryStream ms = new MemoryStream(buff);
            return formatter.Deserialize(ms);
        }

        private static void SerializeObject(Object obj)
        {
            if (obj == null) return;
            BinaryFormatter formatter = new BinaryFormatter();
            using (FileStream ms = new FileStream(@".\Test.bin", FileMode.Create))
            {
                formatter.Serialize(ms, obj);
            }
        }
    }

如果我们将布尔值转换为整数,那将是 0 或 1。你能试试吗:

int val=int.Tryparse(myBooleanValue);

int val= myBooleanValue?1:0;

您可以使用 ISerializable 处理这种情况,但是您需要使用 SerializationInfo.GetEnumerator to determine what type of data was actually read from the stream. In addition, you need to be aware that BinaryFormatter serializes fields not properties, so if your class used Auto-Implemented Properties then the name previously stored in the binary stream will be the name of the backing field 而不是 属性.

的名称来遍历反序列化的属性

例如,假设这是你的原创 class:

[Serializable]
public class Foo
{
    public bool Bar { get; set; }

    public Foo() { }
}

现在您想将 Bar 更改为整数。要序列化或反序列化新旧 BinaryFormatter 流,请使用 ISerializable 的以下实现:

[Serializable]
public class Foo : ISerializable
{
    public int Bar { get; set; }

    public Foo() { }

    public Foo(SerializationInfo info, StreamingContext context)
    {
        var enumerator = info.GetEnumerator();
        while (enumerator.MoveNext())
        {
            var current = enumerator.Current;
            Debug.WriteLine(string.Format("{0} of type {1}: {2}", current.Name, current.ObjectType, current.Value));
            if (current.Name == "Bar" && current.ObjectType == typeof(int))
            {
                Bar = (int)current.Value;
            }
            else if (current.Name == "<Bar>k__BackingField" && current.ObjectType == typeof(bool))
            {
                var old = (bool)current.Value;
                Bar = (old ? 1 : 0); // Or whatever.
            }
        }
    }

    public void GetObjectData(SerializationInfo info, StreamingContext context)
    {
        info.AddValue("Bar", Bar);
    }
}