无法将JSON数组反序列化为类型

时间:2012-02-21 18:19:24

标签: c# json serialization json.net

我正在使用Newtonsoft JSON.NET包来反序列化我的回复。这是来自调用的示例响应,该调用从Omniture中提取可用指标。

[
{"rsid":"somersid",
 "site_title":"somesitetitle",
 "available_metrics":[
    {"metric_name":"averagepagedepth","display_name":"Average Page Depth"},
    {"metric_name":"averagetimespentonpage","display_name":"Average Time Spent on Page"},
    {"metric_name":"averagetimespentonsite","display_name":"Average Time Spent on Site"},
    {"metric_name":"averagevisitdepth","display_name":"Average Visit Depth"},
    {"metric_name":"customersdaily","display_name":"Daily Unique Customers"},
    {"metric_name":"customersloyal","display_name":"Loyal Customers"},
    {"metric_name":"customersmonthly","display_name":"Monthly Unique Customers"},
    {"metric_name":"customersnew","display_name":"New Customers"},
    {"metric_name":"customersquarterly","display_name":"Quarterly Unique Customers"}
    ]
 }
]

对于我想要反序列化的类,我首先尝试了svcutil针对其模式生成的类。它正确地创建了类,并且在反序列化简单对象时它们似乎有效。但是任何具有嵌套数组的内容(如上所述)都会因“无法将JSON数组反序列化为类型”错误而失败。经过仔细检查,svcutil生成的类使用如下数组:

public partial class report_suite_metrics {
    public string rsid { get; set;} ;
    public string site_title { get; set;} ;
    public metric[] available_metrics { get; set;} ;
}

public partial class metric {
    public string metric_name {get; set;}
    public string display_name {get; set;}
}

也许JSON.NET试图将该数组强制转换为我认为的通用List。我看到的所有JSON.NET示例都使用List来处理子数组,因此我创建了新的类来使用json2csharp实用程序(http://json2csharp.com/)进行测试并得到了这个:

public class AvailableMetric {
    public string metric_name { get; set; }
    public string display_name { get; set; }
}

public class RootObject {
    public string rsid { get; set; }
    public string site_title { get; set; }
    public List<AvailableMetric> available_metrics { get; set; }
}

看起来与所有JSON.NET示例完全一致。但它会抛出完全相同的错误:(什么给出?任何人遇到这种问题?这似乎是一个非常简单和常见的情况,所以我很惊讶它不仅仅按预期工作。谢谢

1 个答案:

答案 0 :(得分:2)

你提供的两种类型的工作;问题是返回的源数据是核心类型的数组,而不是单个实例。请注意您在整个数据单元周围提供的JSON示例中的方括号。

这是一个测试控制台应用,可用于检查您看到的JSON。

要使用它,请将您在上面提供的JSON复制到任意文本文件中,然后从应用程序中打开它。

请注意,将反序列化作为数组运行正常,但单个元素(根据您的期望)会失败。

如果您创建一个包含相同样本数据但没有外部[]括号的示例文本文件,您将看到数组反序列化失败并且单个元素有效。

using System.Windows.Forms;
using Newtonsoft.Json;
using Newtonsoft.Json.Serialization;

namespace RptSuiteMetricsTest1
{
    class Program
    {
        public class report_suite_metrics
        {
            public string rsid { get; set; }
            public string site_title { get; set; }
            public metric[] available_metrics { get; set; }
        }

        public class metric
        {
            public string metric_name { get; set; }
            public string display_name { get; set; }
        }

        public class AvailableMetric
        {
            public string metric_name { get; set; }
            public string display_name { get; set; }
        }

        public class RootObject
        {
            public string rsid { get; set; }
            public string site_title { get; set; }
            public List<AvailableMetric> available_metrics { get; set; }
        }

        [STAThread]
        static void Main(string[] args)
        {
            OpenFileDialog dlg = new OpenFileDialog();
            DialogResult dr = dlg.ShowDialog();
            if (dr != DialogResult.OK) { return; }

            // NOTE: ...or replace this with a string containing the json data to test.
            string jsonData = System.IO.File.ReadAllText(dlg.FileName);

            Console.WriteLine("Deserialize as array");
            try
            {
                report_suite_metrics[] dataset = JsonConvert.DeserializeObject<report_suite_metrics[]>(jsonData);
                Console.WriteLine("{0} items found.", dataset.Length);
            }
            catch (Exception ex)
            {
                Console.WriteLine("Exception of type {0}", ex.GetType().Name);
            }

            Console.WriteLine("Deserialize as array, alternate type");
            try
            {
                RootObject[] dataset = JsonConvert.DeserializeObject<RootObject[]>(jsonData);
                Console.WriteLine("{0} items found.", dataset.Length);
            }
            catch (Exception ex)
            {
                Console.WriteLine("Exception of type {0}", ex.GetType().Name);
            }

            Console.WriteLine();

            Console.WriteLine("Deserialize as single item");
            try
            {
                report_suite_metrics dataset = JsonConvert.DeserializeObject<report_suite_metrics>(jsonData);
                Console.WriteLine("rsid=\"{0}\"", dataset.rsid);
            }
            catch (Exception ex)
            {
                Console.WriteLine("Exception of type {0}", ex.GetType().Name);
            }

            Console.WriteLine("Deserialize as single item, alternate type");
            try
            {
                RootObject dataset = JsonConvert.DeserializeObject<RootObject>(jsonData);
                Console.WriteLine("rsid=\"{0}\"", dataset.rsid);
            }
            catch (Exception ex)
            {
                Console.WriteLine("Exception of type {0}", ex.GetType().Name);
            }

            Console.ReadKey();
        }
    }
}