使用空数组反序列化JSON对象

时间:2016-02-12 08:10:16

标签: c# json json.net json-deserialization

当有一个空/ null属性(使用JSON.NET)时,我在反序列化JSON对象时遇到一些麻烦,并希望有人能指出我正确的方向。以下是我正在尝试的代码片段,并在dotnetfiddle

进行测试

这是JSON的一个示例:

{
    "`LCA0009": [],
    "`LCA0001": {
        "23225007190002": "1",
        "23249206670003": "1",
        "01365100070018": "5"
    },
    "`LCA0003": {
        "23331406670018": "1",
        "24942506670004": "1"
    },
    "`LCA0005": {
        "01365100070018": "19"
    }
}

我试图使用此代码:

using System;
using System.Collections.Generic;
using Newtonsoft.Json;

public class Program
{
    public static void Main()
    {

        string json = "{\"`LCA0009\": [], \"`LCA0001\": {\"23225007190002\": \"1\",\"23249206670003\": \"1\",\"01365100070018\": \"5\"},\"`LCA0003\": {\"23331406670018\": \"1\",\"24942506670004\": \"1\"},\"`LCA0005\": {\"01365100070018\": \"19\"}}";
        Console.WriteLine(json);
        Console.WriteLine();

        Console.WriteLine("This works");
        var root = JsonConvert.DeserializeObject(json);
        Console.WriteLine(root);

        Console.WriteLine("This doesn't work");
        var root2 = JsonConvert.DeserializeObject<Dictionary<string, Dictionary<string, int>>>(json);
        Console.WriteLine(root2);

        foreach (var locationKvp in root2)
        {
            foreach (var skuKvp in locationKvp.Value)
            {
                Console.WriteLine("location: " + locationKvp.Key + ", sku: " +  skuKvp.Key + ", qty: " + skuKvp.Value);
            }
        }
    }
}

&#34;不起作用&#34;上面是我得到这个错误:

  

运行时异常(第19行):无法将当前JSON数组(例如[1,2,3])反序列化为类型&System; System.Collections.Generic.Dictionary`2 [System.String,System。 INT32]&#39;因为类型需要JSON对象(例如{&#34; name&#34;:&#34; value&#34;})才能正确反序列化。   要修复此错误,请将JSON更改为JSON对象(例如{&#34; name&#34;:&#34; value&#34;})或将反序列化类型更改为实现集合接口的数组或类型(例如ICollection,IList)就像可以从JSON数组反序列化的List。 JsonArrayAttribute也可以添加到类型中以强制它从JSON数组反序列化。   路径&#39;`LCA0009&#39;,第1行,第14位。

如何删除具有null / empty数组的属性?

1 个答案:

答案 0 :(得分:3)

从根本上说,JSON是不一致的,最好的答案是使其保持一致。无论谁给你JSON需要被告知要解决它。

除非你能做到,否则你可以转换&#34; raw&#34;版本为Dictionary<string, Dictionary<string, int>>你自己,但它的工作量更多:

https://dotnetfiddle.net/zdeOOX

string json = "{\"`LCA0009\": [], \"`LCA0001\": {\"23225007190002\": \"1\",\"23249206670003\": \"1\",\"01365100070018\": \"5\"},\"`LCA0003\": {\"23331406670018\": \"1\",\"24942506670004\": \"1\"},\"`LCA0005\": {\"01365100070018\": \"19\"}}";

// Convert it
var root = (JObject)JsonConvert.DeserializeObject(json);
var results = new Dictionary<string, Dictionary<string,int>>();
foreach (var entry in root)
{
    var dict = new Dictionary<string,int>();
    if (!(entry.Value is JArray))
    {
        foreach (var subentry in (JObject)entry.Value)
        {
            int v;
            if (int.TryParse(((JValue)subentry.Value).ToString(), out v))
            {
                dict.Add(subentry.Key, v);
            }
        }
    }
    results.Add(entry.Key, dict);
}

// Results:
foreach (var name in results.Keys)
{
    var entry = results[name];
    Console.WriteLine(name + ":");
    foreach (var entryKey in entry.Keys)
    {
        Console.WriteLine("- " + entryKey + ": " + entry[entryKey]);
    }
}

我希望Linq能够做得更优雅。