将 JSON 反序列化为自定义列表

本文关键字:自定义 列表 反序列化 JSON | 更新日期: 2023-09-27 17:56:48

我有这个json:

var x = [
    [99,"abc","2dp",{"GroupNum": 0,"Total":[4, 1]}],
    [7,"x","date"],
    [60,"x","1dp",{"GroupNum": 1}],
    ...
]

存在以下规则(让我们参考内部列表索引i):

  • x[i][0] - 必填项 - 始终为整数
  • x[i][1] - 必填项 - 始终为字符串
  • x[i][2] - 必填项 - 始终为字符串
  • x[i][3] - 可选项 - 如果存在,则具有以下规则:
    • x[i][3].GroupNum - 必填字段 - 始终为整数
    • x[i][3].Total - 可选字段 - 当它存在时,它是一个整数列表

因此,我为这些规则创建了以下类:

public class ReportTemplateField : System.Collections.CollectionBase
{
    public object this[int index]
    {
        get
        {
            switch (index)
            {
                case 0:
                    return (int)List[index];
                case 1:
                case 2:
                    return (string)List[index];
                case 3:
                    return (ReportGrouping)List[index];
                default:
                    throw new System.ArgumentOutOfRangeException("Class ReportTemplateField only contains 4 items");
            }
        }
        set
        {
            switch (Count)
            {
                case 0:
                    List[index] = unchecked((int)value);
                    break;
                case 1:
                case 2:
                    List[index] = (string)value;
                    break;
                case 3:
                    List[index] = (ReportGrouping)value;
                    break;
                default:
                    throw new System.ArgumentOutOfRangeException("Class ReportTemplateField may only contain 4 items");
            }
        }
    }
    public void Add(object item)
    {
        switch (Count)
        {
            case 0:
                List.Add(unchecked((int)item));
                break;
            case 1:
            case 2:
                List.Add((string)item);
                break;
            case 3:
                List.Add((ReportGrouping)item);
                break;
            default:
                throw new System.ArgumentOutOfRangeException("Class ReportTemplateField may only contain 4 items");
        }
    }
}
public class ReportGrouping
{
    public int GroupNum { get; set; }
    public List<int> Total { get; set; }
}

但是,当我尝试使用 json.net 反序列化时,ReportTemplateField 类永远不会被使用。我知道这一点,因为我在每个方法中都放置了断点,并且没有一个被触发:

string json = "[[99,'"abc'",'"2dp'",{'"GroupNum'": 0,'"Total'":[4, 1]}],[7,'"x'",'"date'"],[60,'"x'",'"1dp'",{'"GroupNum'": 1}]]";
List<ReportTemplateField> parsed = JsonConvert.DeserializeObject<List<ReportTemplateField>>(json);
int Total01 = parsed[0][3].Total[1];

给我这个错误:

错误 CS1061:"对象"不包含"总计"的定义,并且 没有扩展方法"总计"接受类型的第一个参数 可以找到"对象"(您是否缺少使用指令或 程序集引用?

但是,如果我只是单独反序列化ReportGrouping,它可以工作:

string json2 = "{'"GroupNum'": 0,'"Total'":[4, 1]}";
ReportGrouping parsed2 = JsonConvert.DeserializeObject<ReportGrouping>(json2);
parsed2.Total[1]; // 1

所以问题出在ReportTemplateField类上。也许我不应该继承System.Collections.CollectionBase?我基本上按照这个演练进行了一些修改 - https://msdn.microsoft.com/en-us/library/xth2y6ft(v=vs.71).aspx但它确实说文档可能已经过时了。我使用的是 .NET 4.5,所以请在您的任何答案中考虑这一点。

如何让 Json.net 正确反序列化,以便我可以parsed[0][3].Total[1];访问第二个总数的整数值,而无需强制转换它或对它执行任何操作?

将 JSON 反序列化为自定义列表

代码中有一些错误,1 是对象没有名为 Total 的属性,另一个是 parsed[0][3] 是一个字符串。

代码中有一些错误:

  1. parsed[0][3] 是一个字符串。通过将字符串转换为报表分组来修复它。
  2. 新的报告分组()。组名应根据 JSON 命名为组。
  3. 将 parsed[0][3] 解析为 ReportingGroup 后,它仍然会返回一个对象,因此之后将 i 转换为 ReportingGroup。 即(报告组)解析[0][3]。

    get
    {
        switch (index)
        {
            case 0:
                return (int)List[index];
            case 1:
            case 2:
                return (string)List[index];
            case 3:
                return JsonConvert.DeserializeObject<ReportGrouping>(List[index].ToString());
            default:
                throw new System.ArgumentOutOfRangeException("Class ReportTemplateField only contains 4 items");
        }
    }
    

并像这样使用它:

List<ReportTemplateField> parsed = JsonConvert.DeserializeObject<List<ReportTemplateField>>(json);
ReportGrouping group = (ReportGrouping)parsed[0][3];

不要忘记将组名重命名为组并修复二传手。

编辑:很抱歉重复了这么多次。