我正在尝试将反序列化移植到protobuf-net。但我无法超越这一点:
[DataContract]
public abstract class BaseObject : IBaseObject, INotifyPropertyChanged
{
// only methods and events here
}
[DataContract]
public abstract class BaseTableContract<TSelf> : BaseObject, ITableContract<TSelf>
where TSelf : BaseTableContract<TSelf>, new()
{
// only methods and events here
}
[DataContract]
public abstract class BaseTableSharedContract<TSelf, TIdentifier> : BaseTableContract<TSelf>, ITableSharedContract<TSelf, TIdentifier>
where TSelf : BaseTableSharedContract<TSelf, TIdentifier>, new()
where TIdentifier : IComparable
{
[DataMember(Order = 1)] public TIdentifier ID { get; set; }
[DataMember(Order = 2)] public DateTime Inserted { get; set; }
[DataMember(Order = 3)] public DateTime? Updated { get; set; }
}
[DataContract]
public abstract class BaseDataTableContract<TSelf> : BaseTableSharedContract<TSelf, Guid>, IDataTableContract<TSelf>
where TSelf : BaseDataTableContract<TSelf>, new()
{
[DataMember(Order = 4)] public Guid IdUserInserted { get; set; }
[DataMember(Order = 5)] public Guid? IdUserUpdated { get; set; }
[DataMember(Order = 6)] public String Description { get; set; }
}
[DataContract]
public class FinalContract : BaseDataTableContract<FinalContract>
{
[DataMember(Order = 7)] public Guid SomeProperty { get; set; }
}
然后我在运行时(客户端/服务器)中注册继承的结构(很早在进行任何序列化之前):
MetaType metaBase = RuntimeTypeModel.Default[typeof(BaseObject)];
metaBase.UseConstructor = false;
MetaType metaTable = metaBase.AddSubType(1, typeof(BaseTableContract<FinalContract>);
metaTable.UseConstructor = false;
MetaType metaShared = metaTable.AddSubType(2, typeof(BaseTableSharedContract<FinalContract, Guid>);
metaShared.UseConstructor = false;
MetaType metaData = metaShared.AddSubType(3, typeof(BaseDataTableContract<FinalContract>);
metaData.UseConstructor = false;
metaData.AddSubType(4, typeof(FinalContract));
如果没有将UseConstructor
属性设置为false
,则会抛出它找不到无参数构造函数的情况。没错,没有显式构造函数(C#自动创建的构造函数除外)。因此我将其关闭(false),但仍然会反序列化:
stream.Seek(0, SeekOrigin.Begin);
TResult result = ProtoBuf.Serializer.Deserialize<TResult>(stream);
例外:无法创建抽象类
似乎可以进行序列化(17kB od数据)。我可以在反序列化之前在RuntimeTypeModel中检查继承结构。我在这里做什么错了?
答案 0 :(得分:0)
问题就在我们这边-可能是。我们正在计算完整类型名称(字符串)的正数哈希(整数),以用作子类型唯一代码(在AddSubType()
方法中)。这某种程度上是行不通的。如果将其替换为从100开始索引的数字(例如),它将起作用。
我们通过尝试编译我给出的示例而发现了它,并且它起作用了(在进行较小的更改后才可以编译)。当我们使用完整的类层次结构1:1时,会引发该错误。在我们注释掉除DataMember
属性之外的所有内部内容之后。仅在将哈希方法更改为线性序列后,它才起作用。当然,现在服务器和客户端包含相同的加载顺序是敏感的。但这不再是问题。
我希望这会对类似情况的人有所帮助。