Protobuf-net:无法创建摘要 class

Protobuf-net: Cannot create an abstract class

我正在尝试将反序列化移植到 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; }
}

然后我在 运行 时间(两个 client/server)注册继承的结构(在任何序列化发生之前),如下所示:

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

异常:无法创建摘要class

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

问题出在我们这边 - 可能。我们正在计算要用作子类型唯一代码(在 AddSubType() 方法中)的完整类型名称(字符串)的正数哈希 (int)。这在某种程度上是行不通的。当替换为从 100 开始索引的数字(例如)时,它有效。

我们通过尝试编译我给出的示例发现它有效(经过微小更改后可编译)。当我们使用完整的 class 层次结构 1:1 时,它会抛出该错误。我们注释掉里面除了DataMember属性之外的所有内容之后。只有在将散列方法更改为线性序列后它才起作用。当然,现在服务器和客户端包含相同的加载顺序是很敏感的。但这不再是问题。

我希望这对处于类似情况的人有所帮助。