27

给定以下json:

[ {"id":"123", ... "data":[{"key1":"val1"}, {"key2":"val2"}], ...}, ... ]

那是更大树的一部分,我如何将“数据”属性反序列化为:

List<MyCustomClass> Data { get; set; }

或者

List<KeyValuePair> Data { get; set; }

或者

Dictionary<string, string> Data { get; set; }

使用 Json.NET?任何一个版本都可以(尽管我更喜欢 List of MyCustomClass)。我已经有一个包含其他属性的类,如下所示:

public class SomeData
{
   [JsonProperty("_id")]
   public string Id { get; set; }
   ...
   public List<MyCustomClass> Data { get; set; }
}

其中“MyCustomClass”将仅包含两个属性(键和值)。我注意到有一个 KeyValuePairConverter 类听起来可以满足我的需要,但我找不到如何使用它的示例。谢谢。

4

4 回答 4

33

最简单的方法是将键值对数组反序列化为IDictionary<string, string>


public class SomeData
{
    public string Id { get; set; }

    public IEnumerable<IDictionary<string, string>> Data { get; set; }
}

private static void Main(string[] args)
{
    var json = "{ \"id\": \"123\", \"data\": [ { \"key1\": \"val1\" }, { \"key2\" : \"val2\" } ] }";

    var obj = JsonConvert.DeserializeObject<SomeData>(json);
}

但是,如果您需要将其反序列化为您自己的类,它可能看起来像这样:


public class SomeData2
{
    public string Id { get; set; }

    public List<SomeDataPair> Data { get; set; }
}

public class SomeDataPair
{
    public string Key { get; set; }

    public string Value { get; set; }
}

private static void Main(string[] args)
{
    var json = "{ \"id\": \"123\", \"data\": [ { \"key1\": \"val1\" }, { \"key2\" : \"val2\" } ] }";

    var rawObj = JObject.Parse(json);

    var obj2 = new SomeData2
    {
        Id = (string)rawObj["id"],
        Data = new List<SomeDataPair>()
    };

    foreach (var item in rawObj["data"])
    {
        foreach (var prop in item)
        {
            var property = prop as JProperty;

            if (property != null)
            {
                obj2.Data.Add(new SomeDataPair() { Key = property.Name, Value = property.Value.ToString() });
            }

        }
    }
}

看到我知道那Valuestring并且我调用ToString()方法,可以有另一个复杂的类。

于 2013-04-03T15:11:27.563 回答
2

感谢@Boo 的回答,但就我而言,我需要进行一些小的调整。这就是我的 JSON 的样子:

{
    "rates": {
        "CAD": 1.5649,
        "CZK": 26,118,
        ...
    },
    "base": "EUR",
    "date": "2020-08-16"
}

我的 DTO 如下所示:

public IDictionary<string, decimal> Rates { get; set; }
public string Base { get; set; }
public DateTime Date { get; set; }

所以唯一的调整是删除IEnumerable周围的IDictionary

于 2020-08-15T23:31:06.837 回答
1

我最终这样做了:

[JsonConverter(typeof(MyCustomClassConverter))]
public class MyCustomClass
{
  internal class MyCustomClassConverter : JsonConverter
  {
    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
      throw new NotImplementedException();
    }

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
      JObject jObject = JObject.Load(reader);

      foreach (var prop in jObject)
      {
        return new MyCustomClass { Key = prop.Key, Value = prop.Value.ToString() };
      }

      return null;
    }

    public override bool CanConvert(Type objectType)
    {
      return typeof(MyCustomClass).IsAssignableFrom(objectType);
    }
  }

  public string Key { get; set; }
  public string Value { get; set; }
}
于 2013-04-03T17:36:54.473 回答
0
public class Datum
{
    public string key1 { get; set; }
    public string key2 { get; set; }
}

public class RootObject
{
    public string id { get; set; }
    public List<Datum> data { get; set; }
}

我也使用了这个向导json2csharp.com来生成反序列化的类

使用它

using RestSharp;
using Newtonsoft.Json;

IRestResponse restSharp= callRestGetMethodby_restSharp(api_server_url);
string jsonString= restSharp.Content;

RootObject rootObj= JsonConvert.DeserializeObject<RootObject>(jsonString);
return Json(rootObj);

如果你用 restsharp 打电话休息

    public IRestResponse callRestGetMethodby_restSharp(string API_URL)
    {
        var client = new RestSharp.RestClient(API_URL);
        var request = new RestRequest(Method.GET);
        request.AddHeader("Content-Type", "application/json");
        request.AddHeader("cache-control", "no-cache");
        IRestResponse response = client.Execute(request);
        return response;
    }

您也可以从getpostman.com工具中获得这 6 行 restsharp

于 2018-07-31T23:42:35.453 回答