使用Newtonsoft在C#中反序列化嵌套的JSON对象

5

这是一个我尝试反序列化的示例JSON字符串:

{
    "background": "#ededf0",
    "theme": "Flat",
    "name": "Control Page Name",
    "scriptName": "Control Page Script",
    "objects": [{
        "ID": "76799598",
        "position": {
            "top": 0.30428571428571427,
            "left": 0.6054421768707483
        },
        "width": 0.09451596023024594,
        "height": 0.07450128571428571,
        "label": "btn",
        "colour": "blue",
        "action": "OpenLayout",
        "actionParams": {
            "actionName": "Explorer",
            "itemId": "91d5bfff-a723-498a-a846-24a9e41fbaa6",
            "groupId": "bf434b0d-90c2-496a-96ce-c09f228255b4"
        }
    }]
}

我似乎无法遍历JSON的“objects”键节点值。 在“background”,“theme”,“name”和“scriptName”之前的所有内容都可以正常工作。 这是我的转换器

{
        JObject panel = (JObject)serializer.Deserialize(reader);

        var cpResponse = new VWControlPanelResponse();

        JToken scriptGroupId, scriptId;
        cpResponse.Background = (string)((JValue)panel["background"]).Value;
        cpResponse.Theme = (string)((JValue)panel["theme"]).Value;
        cpResponse.Name = (string)((JValue)panel["name"]).Value;
        cpResponse.ScriptName = (string)((JValue)panel["scriptName"]).Value;
        cpResponse.ScriptGroupId = panel.TryGetValue("scriptGroupId", out scriptGroupId) ? new Guid(scriptGroupId.ToString()) : Guid.Empty;
        cpResponse.ScriptId = panel.TryGetValue("scriptId", out scriptId) ? new Guid(scriptId.ToString()) : Guid.Empty;

        cpResponse.VisualElements = serializer.Deserialize<List<VisualElement>>(reader);


        return cpResponse;
    }

...这是我的模型:

[JsonConverter(typeof(Converter))]
public class Response
{
    public string Background { get; set; }
    public string Theme { get; set; }
    public string Name { get; set; }
    public string ScriptName { get; set; }

    public Guid ScriptGroupId { get; set; }
    public Guid ScriptId { get; set; }

    public List<VisualElement> VisualElements { get; set; }

}

public class VisualElement
{
    public long ID { get; set; }
}

我已经阅读了很多类似的在线文章(特别是JSON Deserialization C#),但我无法弄清楚为什么它不想翻译我的VisualElements节点。 我尝试使用reader.Read()操作,但读取器的标记类型表明它是对象的末尾。


真的吗?你的JSON数据都在一个超长行里面?谁会去读那个啊? - user2819245
1
编辑了你的问题并以更可读的方式格式化了 JSON 数据块... - user2819245
只是为了解释导致您问题的原因(请使用L.B.的答案作为解决方案):请注意,您已经使用serializer.Deserialize的第一次调用对根JSON对象及其所有内部对象(包括“objects”集合)进行了反序列化。当该方法返回时,读取器的位置位于JSON数据的末尾。因此,serializer.Deserialize的第二次调用将不再提供任何数据... - user2819245
1个回答

12

你可以将你的类声明为

public class Position
{
    public double top { get; set; }
    public double left { get; set; }
}

public class ActionParams
{
    public string actionName { get; set; }
    public string itemId { get; set; }
    public string groupId { get; set; }
}

public class VisualElement
{
    public string ID { get; set; }
    public Position position { get; set; }
    public double width { get; set; }
    public double height { get; set; }
    public string label { get; set; }
    public string colour { get; set; }
    public string action { get; set; }
    public ActionParams actionParams { get; set; }
}

public class Response
{
    public string background { get; set; }
    public string theme { get; set; }
    public string name { get; set; }
    public string scriptName { get; set; }
    public List<VisualElement> objects { get; set; }
}

反序列化
 var response = JsonConvert.DeserializeObject<Response>(json);

并将其用作

foreach (var vObj in response.objects)
{
    Console.WriteLine(vObj.colour);
}

感谢您的输入,我过去常常这么做,并且它很好用,但是理想情况下,我希望集成JSON转换器,因为我想将位置值(top,left)转换为包含这两个分数的自定义区域对象。 - Dramos
2
@Dramos,如果我不知道你的自定义JsonConverter是做什么的,以及你的Region对象是什么,你希望我怎么做呢? - L.B
没关系,抱歉问题有误导性。我只是想了解为什么我无法在转换器中迭代我的“objects”节点。elgonzo给了我答案(“读取器的位置在JSON数据的末尾。因此,不会再次读取和提供数据。”) - Dramos

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接