Protobuf-net 泛型继承和封闭构造泛型类型

sho*_*mat 3 protobuf-net

我有一个相当复杂的继承层次结构,包括泛型,我们正在尝试使用 protobuf .net 进行序列化。不幸的是,它似乎无法正确处理这种情况。这就是层次结构的样子。

    [System.Runtime.Serialization.DataContract]
    [ProtoBuf.ProtoInclude(1000, typeof(GenericBaseClass<object>))]
    [ProtoBuf.ProtoInclude(1001, typeof(GenericBaseClass<string>))]
    public abstract class BaseClass
    {

        public int BaseProperty1 { set; get; }
        public int BaseProperty2 { set; get; }

        public BaseClass()
        {

        }

    }

    [System.Runtime.Serialization.DataContract]
    [ProtoBuf.ProtoInclude(1002, typeof(GenericDerivedClass<object>))]
    [ProtoBuf.ProtoInclude(1003, typeof(GenericDerivedClass<string>))]
    public abstract class GenericBaseClass<T> : BaseClass
    {
        /// <summary>
        /// 
        /// </summary>
        [System.Runtime.Serialization.DataMember(Order = 5)]
        public T ResponseProperty
        {
            get;
            set;
        }

        public GenericBaseClass()
        {
        }
    }

    [System.Runtime.Serialization.DataContract]
    [ProtoBuf.ProtoInclude(1004, typeof(DerivedClass1))]
    [ProtoBuf.ProtoInclude(1005, typeof(DerivedClass2))]
    public abstract class GenericDerivedClass<T> : GenericBaseClass<T>
    {
        public int AdditionalProperty { get; set; }

        public GenericDerivedClass()
        {

        }
    }
Run Code Online (Sandbox Code Playgroud)

最后这些类由两个封闭构造的非泛型类实现

    [System.Runtime.Serialization.DataContract]
    public class DerivedClass1 : GenericDerivedClass<string>             
    {
        [System.Runtime.Serialization.DataMember(Order = 6)]
        public int DerivedClass1Property { set; get; }
    }

    [System.Runtime.Serialization.DataContract]
    public class DerivedClass2 : GenericDerivedClass<object>
    {
        [System.Runtime.Serialization.DataMember(Order = 7)]
        public int DerivedClass2Property { set; get; }
    }
Run Code Online (Sandbox Code Playgroud)

我编写了以下测试方法来序列化这些方法,但它给了我错误。

    [TestMethod]
    public void SerializeDeserializeAndCompare()
    {            

        DerivedClass2 i = new DerivedClass2() { BaseProperty1 = 1, BaseProperty2 = 2, DerivedClass2Property = 3, ResponseProperty = new Object() };
        using (var file = System.IO.File.Create("test.bin"))
        {
            ProtoBuf.Serializer.Serialize(file, i);
        }

        using (var file = System.IO.File.OpenRead("test.bin"))
        {
            var o = ProtoBuf.Serializer.Deserialize<DerivedClass2>(file);
        }
    }
Run Code Online (Sandbox Code Playgroud)

我得到的错误是

ProtoBuf.ProtoException:一种类型只能参与一个继承层次结构 (CapitalIQ.DataGet.UnitTests.DataSetUnitTest+DerivedClass2) ---> System.InvalidOperationException:一种类型只能参与一个继承层次结构

这是 protobuf .net 的限制还是我做错了什么。我用的是r282版本。

谢谢肖比特

Mar*_*ell 5

与所有属性一样,属性中包含的类型信息适用于泛型类型定义中的所有封闭类型。因此,您实际定义的(对 protobuf-net)是:

BaseClass
: GenericBaseClass<object>
 : GenericDerivedClass<object>
  : DerivedClass1
  : DerivedClass2
 : GenericDerivedClass<string>
  : DerivedClass1
  : DerivedClass2
: GenericBaseClass<string>
 : GenericDerivedClass<object>
  : DerivedClass1
  : DerivedClass2
 : GenericDerivedClass<string>
  : DerivedClass1
  : DerivedClass2
Run Code Online (Sandbox Code Playgroud)

正如您所看到的,有很多重复项 - 这显然令人困惑。由于属性参数不能使用类型参数,因此可以选择添加某种奇怪的谓词机制,这在我看来非常令人困惑。IMO,手动建模(删除属性ProtoInclude)会是一个更好的主意。我怀疑你想要的模型是:

BaseClass
: GenericBaseClass<object>
 : GenericDerivedClass<object>
  : DerivedClass2
: GenericBaseClass<string>
 : GenericDerivedClass<string>
  : DerivedClass1
Run Code Online (Sandbox Code Playgroud)

protobuf-net 可以使用它,但要解释该模型需要“v2”和RuntimeTypeModel

另请注意,这object对于 protobuf 来说是一个问题;protobuf-net 可以使用动态类型选项来伪造它,但这......并不理想。它当然不能序列化 an object,因此在测试中我替换了一个字符串。另请注意BaseProperty1, 、BaseProperty2AdditionalProperty当前未标记为序列化,但可以很简单。

反正:

RuntimeTypeModel.Default[typeof(BaseClass)]
    .AddSubType(10, typeof(GenericBaseClass<object>))
    .AddSubType(11, typeof(GenericBaseClass<string>));

RuntimeTypeModel.Default[typeof(GenericBaseClass<object>)]
    .AddSubType(10, typeof(GenericDerivedClass<object>));
RuntimeTypeModel.Default[typeof(GenericBaseClass<object>)][5].DynamicType = true; // object!
RuntimeTypeModel.Default[typeof(GenericDerivedClass<object>)]
    .AddSubType(10, typeof(DerivedClass2));

RuntimeTypeModel.Default[typeof(GenericBaseClass<string>)]
    .AddSubType(10, typeof(GenericDerivedClass<string>));
RuntimeTypeModel.Default[typeof(GenericDerivedClass<string>)]
    .AddSubType(10, typeof(DerivedClass1));

DerivedClass2 i = new DerivedClass2() { BaseProperty1 = 1, BaseProperty2 = 2, DerivedClass2Property = 3, ResponseProperty = "some string" };
using (var file = System.IO.File.Create("test.bin"))
{
    ProtoBuf.Serializer.Serialize(file, i);
}

using (var file = System.IO.File.OpenRead("test.bin"))
{
    var o = ProtoBuf.Serializer.Deserialize<DerivedClass2>(file);
}
Run Code Online (Sandbox Code Playgroud)

您不必使用RuntimeTypeModel.Default- 事实上,我建议使用(并缓存)单独的类型模型;但Serializer.Serialize指向默认模型。如果您创建自定义模型 ( TypeModel.Create),只需将其存储在某处并Serialize从那里使用等。