我有一个相当复杂的继承层次结构,包括泛型,我们正在尝试使用 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版本。
谢谢肖比特
与所有属性一样,属性中包含的类型信息适用于泛型类型定义中的所有封闭类型。因此,您实际定义的(对 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
, 、BaseProperty2
和AdditionalProperty
当前未标记为序列化,但可以很简单。
反正:
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
从那里使用等。
归档时间: |
|
查看次数: |
3291 次 |
最近记录: |