我正在使用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示例完全一致。但它会抛出完全相同的错误:(什么给出?任何人遇到这种问题?这似乎是一个非常简单和常见的情况,所以我很惊讶它不仅仅按预期工作。谢谢
答案 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();
}
}
}