1

我得到一个ProtoException

ProtoBuf.ProtoException : Internal error; a key mismatch occurred

使用以下代码:

[ProtoContract]
class Foo { }

class MemberRemovedTest
{
    [ProtoContract]
    class V1
    {
        [ProtoMember(1, AsReference = true)]
        public Foo A { get; set; }

        [ProtoMember(2, AsReference =  true)]
        public Foo B { get; set; }
    }

    [ProtoContract]
    class V2
    {
        [ProtoMember(2, AsReference = true)]
        public Foo B { get; set; }
    }

    public void BasicTest()
    {
        var v1 = new V1();
        v1.A = new Foo();
        v1.B = new Foo();

        byte[] buffer;
        V2 v2;
        using (var stream = new MemoryStream())
        {
            Serializer.Serialize(stream, A);
            buffer = stream.ToArray();
        }
        using (var stream = new MemoryStream(buffer))
        {
            v2 = Serializer.Deserialize<V2>(stream); //Exception here
        }
    }
}

如果出现以下情况,它不会抛出异常:

  • 第二个成员是被删除的成员。
  • or are not的ProtoMember属性。ABAsReference = true
  • A并且B都不是用Foo实例设置的。

我理解 protobuf 支持成员删除,但这似乎表明在某些情况下必须保留它们。

这是 Protobuf 中的错误还是关于删除成员的错误假设?

异常调用堆栈:

   at ProtoBuf.NetObjectCache.SetKeyedObject(Int32 key, Object value) in c:\Dev\protobuf-net\protobuf-net\NetObjectCache.cs: line 67
   at ProtoBuf.BclHelpers.ReadNetObject(Object value, ProtoReader source, Int32 key, Type type, NetObjectOptions options) in c:\Dev\protobuf-net\protobuf-net\BclHelpers.cs: line 425
   at proto_6(Object, ProtoReader)
   at ProtoBuf.Serializers.CompiledSerializer.ProtoBuf.Serializers.IProtoSerializer.Read(Object value, ProtoReader source) in c:\Dev\protobuf-net\protobuf-net\Serializers\CompiledSerializer.cs: line 57
   at ProtoBuf.Meta.RuntimeTypeModel.Deserialize(Int32 key, Object value, ProtoReader source) in c:\Dev\protobuf-net\protobuf-net\Meta\RuntimeTypeModel.cs: line 715
   at ProtoBuf.Meta.TypeModel.DeserializeCore(ProtoReader reader, Type type, Object value, Boolean noAutoCreate) in c:\Dev\protobuf-net\protobuf-net\Meta\TypeModel.cs: line 679
   at ProtoBuf.Meta.TypeModel.Deserialize(Stream source, Object value, Type type, SerializationContext context) in c:\Dev\protobuf-net\protobuf-net\Meta\TypeModel.cs: line 580
   at ProtoBuf.Serializer.Deserialize(Stream source) in c:\Dev\protobuf-net\protobuf-net\Serializer.cs: line 77
   at ####.ProtoBuf.MemberRemovedTest.BasicTest() in MemberRemovedTest.cs: line 56
4

1 回答 1

0

嗯....是的,很有趣。该死。使这种情况发挥作用可能会带来极大的问题;如果没有该成员,我们就没有足够的元数据来反序列化该对象——甚至只是知道它是一个可能用作稍后出现的作为引用对象的占位符的对象。存储每个跳过的字段以供以后处理是不切实际的——实际上,在非平凡的情况下,由于技术原因,这是不可能的(如果 as-reference 是一个对象的子对象,可能向下几个级别,被移除为更高级别的成员 - 由于缺少元数据,我们无法处理)。

我现在没有比“是的,那行不通”更好的评论了。烦恼 - 有趣的场景。

于 2013-06-27T04:26:39.663 回答