JSON .NET使用对象键/名称反序列化为复杂类型

时间:2014-10-01 12:24:18

标签: c# json.net

我在我的项目中使用Newtonsoft JSON .NET来处理我的大多数API / JSON,并且通常反序列化或转换为复杂类型以便于使用。问题是,我似乎无法找到一种处理具有动态名称/密钥的对象的好方法(即每次都不同)。

我需要解析的JSON并不完全符合标准,但目前我无权改变它。相关的JSON如下(我没有包括整个结构)。

"orders": {
  "32f5c31d-a851-40cf-a8bb-2fea4bd27777": {
    "orderId": "ord12345",
    "state": "new",
    "sellAmount": "200",
    "paymentMethod": "buyMethod2",
    "email": "example@email.com",
    "pinOrPhonenumber": "180012345",
    "timestamp": "2014-10-01T07:13:33.868Z"
  },
  "123d98d5-19b1-4cb3-8ab3-b991650b134c": {
    "orderId": "ord12346",
    "state": "pending",
    "sellAmount": "1200",
    "paymentMethod": "buyMethod2",
    "email": "example@email.com",
    "pinOrPhonenumber": "180012345",
    "timestamp": "2014-10-01T07:13:33.868Z"
  }
}

目前的代码如下:

var jResponse = Newtonsoft.Json.Linq.JObject.Parse(rResponse.Content);
var jOrders = jResponse["orders"];
foreach (var jChild in jOrders) {
   string sOrderGUID = ""; //The name/key of the object
   var jOrder = jChild.First.ToObject<Order>();
   //Handle the order data, etc
}

上课:

public class Order {

  [JsonProperty("orderId")]
  public string sOrderID { get; set; }

  [JsonProperty("state")]
  public string sState { get; set; }

  [JsonProperty("sellAmount")]
  public decimal nAmount { get; set; }

  [JsonProperty("paymentMethod")]
  public string sPaymentMethod { get; set; }

  [JsonProperty("email")]
  public string sEmail { get; set; }

  [JsonProperty("phonenumber")]
  public string sPhonenumber { get; set; }

  [JsonProperty("paymentMessage")]
  public string sPaymentMessage { get; set; }

  [JsonProperty("timestamp")]
  public DateTime dTimestamp { get; set; }
}

一直在考虑将其转换为JObject并使用“属性”来获取密钥,但如果可能的话,我希望采用更简洁的方法。有什么指针吗?

更新:词典解决方案很好地适用于订单情况,但是当我尝试将相同的方法应用于JSON的另一部分时,我遇到了另一个问题。

"rates": {
    "sek": {
        "name": "Swedish Krona",
        "usd": {
            "name": "US Dollar",
            "rates": {
                "2014-10-01T12:14:26.497Z": 0.138794,
                "2014-10-01T12:17:51.143Z": 0.138794,
                "2014-10-01T12:26:26.827Z": 0.138794,
                "2014-10-01T12:51:03.347Z": 0.138794
            }
        }
    },
    "usd": {
        "name": "US Dollar",
        "sek": {
            "name": "Swedish Krona",
            "rates": {
                "2014-10-01T12:14:28.763Z": 7.20753,
                "2014-10-01T12:17:52.667Z": 7.20753,
                "2014-10-01T12:26:28.477Z": 7.20753,
                "2014-10-01T12:51:04.963Z": 7.20753
            }
        }
    }
}

问题在于我需要收集&#34;名称&#34;价值以及其余的,但货币代码是可变的。考虑使用嵌套字典,但似乎无法捕捉&#34;名称&#34;。

2 个答案:

答案 0 :(得分:2)

是的,字典是一个更好的答案。

var orderWrapper = JsonConvert.DeserializeObject<OrderWrapper>(json);
IEnumerable<Order> orders = result.GetOrders();

public class OrderWrapper
{
    [JsonProperty("orders")] 
    private Dictionary<string, Order> _orders;

    public IEnumerable<Order> GetOrders()
    {
        return _orders.Values;
    }
}

答案 1 :(得分:1)

我试图在下面的行中解析你的JSON

var result = JsonConvert.DeserializeObject<Dictionary<string,  Dictionary<string,Order>>>(jsonString);

在你的json字符串中,你首先得到一个名为&#34; orders&#34;的密钥,如果外部字典匹配字符串密钥,那么内部字典字符串密钥将与订单密钥匹配,值将是解析为订单