Protobuf-net: не удается создать абстрактный класс - PullRequest
0 голосов
/ 04 марта 2019

Я пытаюсь перенести десериализацию на 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);

Исключение: Невозможно создать абстрактный класс

Кажется, сериализация работает (17 КБ данных).Я могу проверить структуру наследования в RuntimeTypeModel перед десериализацией.Что я здесь не так делаю?

1 Ответ

0 голосов
/ 11 марта 2019

Проблема была на нашей стороне - наверное.Мы рассчитываем хэш положительного числа (int) полного имени типа (строки), который будет использоваться в качестве уникального кода подтипа (в методе AddSubType()).Это как-то не работает.При замене числами, индексированными от 100 (например), это работает.

Мы обнаружили, попытавшись скомпилировать приведенный мною пример, и это сработало (после незначительных изменений, которые можно скомпилировать).Когда мы использовали нашу полную иерархию классов 1: 1, это породило эту ошибку.После того, как мы закомментировали все внутренности, кроме DataMember свойств.Только после изменения метода хеширования на линейную последовательность это сработало.Конечно, теперь чувствительно, что сервер и клиент содержат одинаковый порядок загрузки.Но это больше не проблема.

Надеюсь, это поможет кому-то в подобной ситуации.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...