【问题标题】:Protobuf-net: Cannot create an abstract classProtobuf-net:无法创建抽象类
【发布时间】:2019-03-04 16:38:27
【问题描述】:

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

例外:无法创建抽象类

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

【问题讨论】:

    标签: c# serialization protocol-buffers abstract-class protobuf-net


    【解决方案1】:

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

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

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

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2019-04-07
      • 1970-01-01
      相关资源
      最近更新 更多