Использование JsonConvert.DeserializeObject для десериализации Json в класс C# POCO

Вот мой простой User Класс POCO:

/// <summary>
/// The User class represents a Coderwall User.
/// </summary>
public class User
{
    /// <summary>
    /// A User's username. eg: "sergiotapia, mrkibbles, matumbo"
    /// </summary>
    public string Username { get; set; }

    /// <summary>
    /// A User's name. eg: "Sergio Tapia, John Cosack, Lucy McMillan"
    /// </summary>
    public string Name { get; set; }

    /// <summary>
    /// A User's location. eh: "Bolivia, USA, France, Italy"
    /// </summary>
    public string Location { get; set; }

    public int Endorsements { get; set; } //Todo.
    public string Team { get; set; } //Todo.

    /// <summary>
    /// A collection of the User's linked accounts.
    /// </summary>
    public List<Account> Accounts { get; set; }

    /// <summary>
    /// A collection of the User's awarded badges.
    /// </summary>
    public List<Badge> Badges { get; set; }

}

И метод, который я использую для десериализации ответа JSON в User объект (этот фактический вызов JSON здесь):

private User LoadUserFromJson(string response)
{
    var outObject = JsonConvert.DeserializeObject<User>(response);
    return outObject;
}

Это вызывает исключение:

Невозможно десериализовать текущий объект JSON (например, {"name":"value"}) в тип 'System.Collections.Generic.List`1[CoderwallDotNet.Api.Models.Account]', поскольку для типа требуется массив JSON (например, [1,2,3]) для правильной десериализации.

Чтобы исправить эту ошибку, либо измените JSON на массив JSON (например, [1,2,3]), либо измените десериализованный тип так, чтобы это был нормальный тип.NET (например, не примитивный тип, например, целое число, а не тип коллекции, например, массив или список), которые можно десериализовать из объекта JSON. JsonObjectAttribute также можно добавить к типу, чтобы заставить его десериализоваться из объекта JSON. Путь "accounts.github", строка 1, позиция 129.

Никогда ранее не работавший с этим методом DeserializeObject, я застрял здесь.

Я убедился, что имена свойств в классе POCO совпадают с именами в ответе JSON.

Что я могу попробовать десериализовать JSON в этот класс POCO?

9 ответов

Решение

Вот рабочий пример.

Ключевые точки:

  • Декларация Accounts
  • Использование JsonProperty атрибут

,

using (WebClient wc = new WebClient())
{
    var json = wc.DownloadString("http://coderwall.com/mdeiters.json");
    var user = JsonConvert.DeserializeObject<User>(json);
}

-

public class User
{
    /// <summary>
    /// A User's username. eg: "sergiotapia, mrkibbles, matumbo"
    /// </summary>
    [JsonProperty("username")]
    public string Username { get; set; }

    /// <summary>
    /// A User's name. eg: "Sergio Tapia, John Cosack, Lucy McMillan"
    /// </summary>
    [JsonProperty("name")]
    public string Name { get; set; }

    /// <summary>
    /// A User's location. eh: "Bolivia, USA, France, Italy"
    /// </summary>
    [JsonProperty("location")]
    public string Location { get; set; }

    [JsonProperty("endorsements")]
    public int Endorsements { get; set; } //Todo.

    [JsonProperty("team")]
    public string Team { get; set; } //Todo.

    /// <summary>
    /// A collection of the User's linked accounts.
    /// </summary>
    [JsonProperty("accounts")]
    public Account Accounts { get; set; }

    /// <summary>
    /// A collection of the User's awarded badges.
    /// </summary>
    [JsonProperty("badges")]
    public List<Badge> Badges { get; set; }
}

public class Account
{
    public string github;
}

public class Badge
{
    [JsonProperty("name")]
    public string Name;
    [JsonProperty("description")]
    public string Description;
    [JsonProperty("created")]
    public string Created;
    [JsonProperty("badge")]
    public string BadgeUrl;
}

Другой, более упорядоченный подход к десериализации строки JSON в верблюжьей оболочке в POCO-объекте в паскалях заключается в использовании CamelCasePropertyNamesContractResolver.

Это часть пространства имен Newtonsoft.Json.Serialization. Этот подход предполагает, что единственное различие между объектом JSON и POCO заключается в расположении имен свойств. Если имена свойств пишутся по-разному, вам придется прибегнуть к использованию атрибутов JsonProperty для сопоставления имен свойств.

using Newtonsoft.Json; 
using Newtonsoft.Json.Serialization;

. . .

private User LoadUserFromJson(string response) 
{
    JsonSerializerSettings serSettings = new JsonSerializerSettings();
    serSettings.ContractResolver = new CamelCasePropertyNamesContractResolver();
    User outObject = JsonConvert.DeserializeObject<User>(jsonValue, serSettings);

    return outObject; 
}

Вы могли бы создать JsonConverter, Смотрите здесь пример, похожий на ваш вопрос.

Свойство account определяется следующим образом:

"accounts":{"github":"sergiotapia"}

Ваш POCO утверждает это:

public List<Account> Accounts { get; set; }

Попробуйте использовать этот Json:

"accounts":[{"github":"sergiotapia"}]

Массив элементов (который будет отображаться в списке) всегда заключен в квадратные скобки.

Изменить: Account Poco будет что-то вроде этого:

class Account {
    public string github { get; set; }
}

и, возможно, другие свойства.

Редактировать 2: Чтобы не иметь массива, используйте свойство следующим образом:

public Account Accounts { get; set; }

с чем-то вроде примера класса, который я опубликовал в первом редактировании.

В дополнение к принятому ответу, если у вас есть образец текста JSON, вы можете подключить его к этому конвертеру, выбрать параметры и сгенерировать код C#.

Если вы не знаете тип во время выполнения, эта тема выглядит так, как будто она подходит.

динамически десериализовать JSON в любой переданный объект. C#

to fix this error either change the JSON to a JSON array (e.g. [1,2,3]) or change the
deserialized type so that it is a normal .NET type (e.g. not a primitive type like
integer, not a collection type like an array or List) that can be deserialized from a
JSON object.`

Сообщение целиком указывает на возможность сериализации в объект List, но входные данные должны быть списком JSON. Это означает, что ваш JSON должен содержать

"accounts" : [{<AccountObjectData}, {<AccountObjectData>}...],

Где данные AccountObject - это JSON, представляющий ваш объект Account или объект Badge

То, что он сейчас получает, это

"accounts":{"github":"sergiotapia"}

Где account - это объект JSON (обозначенный фигурными скобками), а не массив объектов JSON (массивы обозначены скобками), что вам и нужно. Пытаться

"accounts" : [{"github":"sergiotapia"}]

Для тех, у кого возникла эта проблема, я не видел значение json должным образом. там вы можете проверить классы, которые должны быть сгенерированы, и вернуть ТОЛЬКО ОДИН из этих классов, как только вы прочитаете json в своем коде.

Например, мне нужен объект списка книг, поэтому мой код должен читать только один

      res = await response.Content.ReadAsAsync<BookList>();

Где список книг выглядит примерно так

          public class BookList
    {

        [JsonProperty("data")]
        public IList<Datum> Data { get; set; }
     }

И в этом списке есть меньший книжный класс, который преобразователь назвал Datum (просто книги).

          public class Datum
    {

        [JsonProperty("id")]
        public string Id { get; set; }

        [JsonProperty("isbn")]
        public string Isbn { get; set; }
    }

Опять же, если у вас есть сомнения https://jsonutils.com/https://jsonutils.com/

Может быть поздно, но использование QuickType — самый простой способ сделать это:

https://app.quicktype.io/

Это не совсем то, что я имел в виду. Что вы делаете, если у вас есть универсальный тип, который будет известен только во время выполнения?

public MyDTO toObject() {
  try {
    var methodInfo = MethodBase.GetCurrentMethod();
    if (methodInfo.DeclaringType != null) {
      var fullName = methodInfo.DeclaringType.FullName + "." + this.dtoName;
      Type type = Type.GetType(fullName);
      if (type != null) {
        var obj = JsonConvert.DeserializeObject(payload);
      //var obj = JsonConvert.DeserializeObject<type.MemberType.GetType()>(payload);  // <--- type ?????
          ...
      }
    }

    // Example for java..   Convert this to C#
    return JSONUtil.fromJSON(payload, Class.forName(dtoName, false, getClass().getClassLoader()));
  } catch (Exception ex) {
    throw new ReflectInsightException(MethodBase.GetCurrentMethod().Name, ex);
  }
}
Другие вопросы по тегам