gpt4 book ai didi

Protobuf-net泛型继承与封闭构造泛型

转载 作者:行者123 更新时间:2023-12-01 13:00:18 33 4
gpt4 key购买 nike

我有一个相当复杂的继承层次结构,包括泛型,我们正在尝试使用 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()
{

}
}

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

    [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; }
}

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

    [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);
}
}

我得到的错误是

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

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

谢谢狗屎

最佳答案

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

BaseClass
: GenericBaseClass<object>
: GenericDerivedClass<object>
: DerivedClass1
: DerivedClass2
: GenericDerivedClass<string>
: DerivedClass1
: DerivedClass2
: GenericBaseClass<string>
: GenericDerivedClass<object>
: DerivedClass1
: DerivedClass2
: GenericDerivedClass<string>
: DerivedClass1
: DerivedClass2

如您所见,有很多重复项 - 这显然令人困惑。由于属性参数不能使用类型参数,这将留下添加某种奇怪的谓词机制的选项,这在我看来非常困惑。 IMO,最好手动对此建模(删除 ProtoInclude 属性)。我怀疑您想要的模型是:

BaseClass
: GenericBaseClass<object>
: GenericDerivedClass<object>
: DerivedClass2
: GenericBaseClass<string>
: GenericDerivedClass<string>
: DerivedClass1

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

另请注意,object 对于 protobuf 来说有点问题; protobuf-net 可以使用动态类型选项伪造它,但这并不理想。它当然不能序列化一个object,所以为了测试我已经替换了一个字符串。另请注意,BaseProperty1BaseProperty2AdditionalProperty 当前未标记为序列化,但可以很简单。

无论如何:

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);
}

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

关于Protobuf-net泛型继承与封闭构造泛型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6509188/

33 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com