Ошибка десериализации Newtonsoft JSON.net, где поля в порядке изменения JSON

Это служба WCF, получающая запросы от устройств Android. Тот же запрос работает от устройств Lollipop, а не от устройств из желе, потому что jellybean по-разному создает JSON при создании.

Исключение:

Неожиданный токен при десериализации объекта: String. Путь ‘SearchFilters.config. $ Type’, строка 1, позиция 212.

Не работает Json:

{ "DeviceType": 2, "SearchFilters": { "config": { "$values": [ { "Collection": { "DeviceType": 2 }, "Category": "" } ], "$type": "System.Collections.Generic.List`1[[Yoosh.SharedClasses.YooshConfig, YooshSharedClassesDll]], mscorlib" } }, "RequestingUserId": "66666666-6666-6666-6666-666666666666", "APIKey": "xxx" } 

Рабочий Джон:

 { "APIKey": "xxx", "DeviceType": 2, "RequestingUserId": "66666666-6666-6666-6666-666666666666", "SearchFilters": { "config": { "$type": "System.Collections.Generic.List`1[[Yoosh.SharedClasses.YooshConfig, YooshSharedClassesDll]], mscorlib", "$values": [ { "Category": "", "Collection": { "DeviceType": 2 } } ] } } } 

Некоторые поля находятся в другом порядке. Это единственное различие.

Класс C #:

 public class QueryParameters { BaseParameters m_baseParameters; Guid m_gRequestingUserId; Dictionary m_SearchFilters; [DataMember] public string APIKey { get { return m_baseParameters.APIKey; } set { m_baseParameters.APIKey = value; } } [DataMember] public BaseParameters.YooshDeviceType DeviceType { get { return m_baseParameters.DeviceType; } set { m_baseParameters.DeviceType = value; } } [DataMember] public string DeviceId { get { return m_baseParameters.DeviceId; } set { m_baseParameters.DeviceId = value; } } [DataMember] public Guid RequestingUserId { get { return m_gRequestingUserId; } set { m_gRequestingUserId = value; } } [DataMember] public Dictionary SearchFilters { get { return m_SearchFilters; } set { m_SearchFilters = value; } } } 

Версия Json.net: 6.0.8

Установите JsonSerializerSettings.MetadataPropertyHandling = MetadataPropertyHandling.ReadAhead .

Согласно документации :

Этот пример десериализует JSON с MetadataPropertyHandling, установленным в ReadAhead, так что свойства метаданных не должны находиться в начале объекта.

  string json = @"{ 'Name': 'James', 'Password': 'Password1', '$type': 'MyNamespace.User, MyAssembly' }"; object o = JsonConvert.DeserializeObject(json, new JsonSerializerSettings { TypeNameHandling = TypeNameHandling.All, // $type no longer needs to be first MetadataPropertyHandling = MetadataPropertyHandling.ReadAhead }); 

Обратите внимание, что этот параметр повлияет на производительность .

Наконец, при использовании TypeNameHandling , обратите внимание на это предостережение от документов Newtonsoft :

TypeNameHandling следует использовать с осторожностью, когда ваше приложение десериализует JSON из внешнего источника. Входящие типы должны быть проверены с помощью специального SerializationBinder при десериализации со значением, отличным от None.

Для обсуждения того, почему это может быть необходимо, см. В описании TypeNameHandling в Newtonsoft Json .

  • Использование JSON.NET в качестве стандартного JSON-сериализатора в ASP.NET MVC 3 - возможно ли это?
  • Уничтожить вложенные JSON в объекты C #
  • Могу ли я сериализовать вложенные свойства в мой class за одну операцию с помощью Json.net?
  • JSON.NET Parser * кажется * будет двойной сериализацией моих объектов
  • Игнорирование нулевых полей в Json.net
  • Как преобразовать datatable в строку json с помощью json.net?
  • Использование преобразователей Json.NET для десериализации свойств
  • Использование JSON.NET для возврата ActionResult
  • JSON.Net выдает StackOverflowException при использовании
  • Как я могу разбирать JSON с C #?
  • C # WCF REST. Как вы используете сериализатор JSON.Net вместо стандартного DataContractSerializer?
  • Давайте будем гением компьютера.