Protobuf-net:无法创建抽象类

问题描述 投票:1回答:1

我正在尝试将反序列化移植到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#自动创建的构造函数)。所以我把它关掉了(假),但它仍然会反序列化:

stream.Seek(0, SeekOrigin.Begin);
TResult result = ProtoBuf.Serializer.Deserialize<TResult>(stream);

例外:无法创建抽象类

序列化似乎有效(17 kB数据)。我可以在反序列化之前检查RuntimeTypeModel中的继承结构。我在这做错了什么?

c# serialization protocol-buffers abstract-class protobuf-net
1个回答
0
投票

问题出在我们这边 - 可能。我们正在计算完整类型名称(字符串)的正数哈希(int),以用作子类型唯一代码(在AddSubType()方法中)。这种方式不起作用。当用从100(例如)索引的数字替换时,它可以工作。

我们已经通过尝试编译我给出的示例找到了它并且它起作用(在稍微更改之后可以编译)。当我们使用我们的完整类层次结构1:1时,它会抛出该错误。之后我们评论了除DataMember属性之外的所有内部。只有在将哈希方法更改为线性序列后才能工作。当然,现在服务器和客户端包含相同的加载顺序是敏感的。但这不再是问题了。

我希望这能帮助处于类似情况的人。

© www.soinside.com 2019 - 2024. All rights reserved.