在没有唯一属性名称的情况下反序列化 JSON



我有一个如下所示的 json 字符串

{
"1ST": {
"name": "One",
"symbol": "1st"
},
"2ND": {
"name": "Two",
"symbol": "2nd"
}
}

我试图将其序列化为 C# 对象。看起来它正在创建一个字典,所以我创建了以下结构

public class Response
{
public Dictionary<string, Item> Objects { get; set; }
}
public class Item
{
public string name { get; set; }
public string symbol { get; set; }
}

在序列化期间运行以下内容

response = JsonConvert.DeserializeObject<Response>(jsonString);

它不会在反序列化时引发错误,但我的响应只是返回为空。我错过了什么?

您已经有了正确的基本想法,但是您有一个您并不真正想要的额外Objects属性:您的JSON实际上是字典。您可以将其直接反序列化为Dictionary<string, Item>>。下面是一个示例:

using System;
using System.Collections.Generic;
using System.IO;
using Newtonsoft.Json;
public class Item
{
public string Name { get; set; }
public string Symbol { get; set; }
public override string ToString() => $"{Name}/{Symbol}";
}
public class Test
{
static void Main()
{
var json = File.ReadAllText("test.json");
var dictionary = JsonConvert.DeserializeObject<Dictionary<string, Item>>(json);
foreach (var entry in dictionary)
{
Console.WriteLine($"{entry.Key}: {entry.Value}");
}
}
}

输出:

1ST: One/1st
2ND: Two/2nd

现在,如果您需要Response类型,有多种选择:

  • 使其派生自Dictionary<string, Item>并删除Objects属性。(如果你需要Objects属性,那就不好了。
  • 反序列化为字典,但随后创建一个新的Response对象并自行分配属性。(如果响应是另一种类型的一部分,则不好。
  • 调查是否有某些属性供 Json.NETObjects视为一种"根"属性。

我在最后一个方面有一些运气,但不是完整的解决方案。如果将Objects更改为Dictionary<string, JToken>,则可以将[JsonExtensionData]应用于它,这使它充当任何不匹配属性的默认字典。但是,我还没有找到说服 Json.NET 使用该属性执行适当转换的方法。您可以创建一个字典,每次添加条目时,该字典都会执行从JTokenItem的转换(使用常规 Json.NET 代码),然后将该值添加到另一个字典中 - 但这非常丑陋。这可能只是 Json.NET 无法处理的情况。

建议使用强类型对象,但另一种方法可以使用将在运行时评估的动态类型。

请查看下面的代码。

请注意,我更改了变量名称,因为我们无法在 C# 中定义以数字开头的变量。

using Newtonsoft.Json;
namespace ConsoleApplication1
{
class Program
{
static void Main(string[] args)
{
var jsonString = @"{
'First': {
'name': 'One',
'symbol': '1st'
},
'Second': {
'name': 'Two',
'symbol': '2nd'
}
}";
//If you want to access using strong type.
var response = JsonConvert.DeserializeObject<Response>(jsonString);
var secondSymbol = response.Second.symbol.ToString();
System.Console.WriteLine(secondSymbol);
//If you want to access using dynamic type, will evalute in runtime.
var responseDynamic = JsonConvert.DeserializeObject<dynamic>(jsonString);
var secondSymbolDynamic = responseDynamic.Second.symbol.ToString() ;
System.Console.WriteLine(secondSymbolDynamic);
System.Console.ReadLine();
}
public class Response
{
public Item First { get; set; }
public Item Second { get; set; }
}
public class Item
{
public string name { get; set; }
public string symbol { get; set; }
}
}
}

希望它会有用。 :)

最新更新