如何使用json.net进行json的递归下降?

22

我正在尝试使用json.net解析一个json文件。该文件的格式如下:

{X:
   {
      Title:"foo",
      xxxx:xxxx
   }
}
{Y:
   {ZZ:
        {Title: "bar",...}
    }
}

我试图递归处理所有带有Title属性的对象,但是我对于JToken、JProperty、JContainer、JValue和JObject感到困惑。阅读源代码并没有让我更明白,而且所有的示例都没有帮助。我想要类似以下的东西:

WalkNode(node, Action<Node> action)
{
    foreach(var child in node.Children)
    {
        Action(child);
        WalkNode(child);
    }
}

Parse()
{
   WalkNode(root, n=>
    {
        if(n["Title"] != null)
        {
           ...
        }
    });
}

2
一些问题:上述示例不是有效的JSON。属性X和Y的包含对象是否应该在数组中,还是您打算将X和Y放在同一个包含对象中?另外,在JSON层次结构中是否会有任何数组需要遍历,还是仅嵌套对象和属性? - Brian Rogers
6个回答

34

下面的代码应该非常接近你要找的东西。我假设有一个外部数组,并且数组可以出现在层次结构的任何位置。(如果这不是真的,你可以简化WalkNode方法的代码,但无论哪种方式都应该可以工作。)

下面的代码应该就是你需要的,我假设有一个最外层的数组,同时这个数组可以出现在结构的任何地方。(如果这个假设不成立,你可以稍微简化WalkNode方法的代码,但它应该仍然有效。)

using System;
using Newtonsoft.Json;
using Newtonsoft.Json.Linq;

namespace JsonRecursiveDescent
{
    class Program
    {
        static void Main(string[] args)
        {
            string json =
            @"[
                {
                    ""X"":
                    {
                        ""Title"":""foo"",
                        ""xxxx"":""xxxx""
                    }
                },
                {
                    ""Y"":
                    {
                        ""ZZ"":
                        {
                            ""Title"":""bar"",
                            ""xxxx"":""xxxx""
                        }
                    }
                }
            ]";

            JToken node = JToken.Parse(json);

            WalkNode(node, n =>
            {
                JToken token = n["Title"];
                if (token != null && token.Type == JTokenType.String)
                {
                    string title = token.Value<string>();
                    Console.WriteLine(title);
                }
            });
        }

        static void WalkNode(JToken node, Action<JObject> action)
        {
            if (node.Type == JTokenType.Object)
            {
                action((JObject)node);

                foreach (JProperty child in node.Children<JProperty>())
                {
                    WalkNode(child.Value, action);
                }
            }
            else if (node.Type == JTokenType.Array)
            {
                foreach (JToken child in node.Children())
                {
                    WalkNode(child, action);
                }
            }
        }

    }
}

但是如果我不知道“Title”存在,而我想要所有的属性呢? - rodolfoprado
3
那是一个不同的问题。如果您想发布一个描述您正在尝试做什么的新问题,我很乐意回答它。请确保标记为 json.net - Brian Rogers

23

还需要做类似的事情。 我想提出我的解决方案。它的优点是:

  • not being recursive
  • no callbacks
  • not assuming any internal structure (arrays)
  • decouples tree traversal from the action needed to be executed

    IEnumerable<JToken> AllTokens(JObject obj) {
        var toSearch = new Stack<JToken>(obj.Children());
        while (toSearch.Count > 0) {
            var inspected = toSearch.Pop();
            yield return inspected;
            foreach (var child in inspected) {
                toSearch.Push(child);
            }
        }
    }
    

    Then you can use linq to filter and perform action:

    var tokens = AllTokens(jsonObj);
    var titles = tokens.Where(t => t.Type == JTokenType.Property && ((JProperty)t).Name == "Title");
    

10

我想分享一下我对@BrianRogers WalkNode 方法的小修改,让它更加灵活多用:

private static void WalkNode(JToken node,
                                Action<JObject> objectAction = null,
                                Action<JProperty> propertyAction = null)
{
    if (node.Type == JTokenType.Object)
    {
        if (objectAction != null) objectAction((JObject) node);

        foreach (JProperty child in node.Children<JProperty>())
        {
            if (propertyAction != null) propertyAction(child);
            WalkNode(child.Value, objectAction, propertyAction);
        }
    }
    else if (node.Type == JTokenType.Array)
    {
        foreach (JToken child in node.Children())
        {
            WalkNode(child, objectAction, propertyAction);
        }
    }
}

然后,OP可以这样做:
WalkNode(json, null, prop =>
{
     if (prop.Name == "Title" && prop.Value.Type == JTokenType.String)
     {
         string title = prop.Value<string>();
         Console.WriteLine(title);
     }
});

8
您也可以使用JSONPath进行操作:node.SelectTokens("$..*"); 使用方法如下:
var jObjectsWithTitle = node
    .SelectTokens("$..*")
    .OfType<JObject>()
    .Where(x => x.Property("Title") != null);

或者只需:
var jObjectsWithTitle = node.SelectTokens("$..[?(@.Title)]");

1

尝试这种方法,我在几次不成功的尝试之后写出来:

 private void Traverse(JToken token, TreeNode tn)
    {
        if (token is JProperty)
            if (token.First is JValue)
                tn.Nodes.Add(((JProperty)token).Name + ": " + ((JProperty)token).Value);
            else
                tn = tn.Nodes.Add(((JProperty)token).Name);

        foreach (JToken token2 in token.Children())
            Traverse(token2, tn);
    }

首先,您需要像这样传递完整的JSON文件:

TreeNode rooty= tvu.Nodes.Add("Rooty") // not the Indian bread,just Rooty,  Ok?
JToken token = JToken.Parse(File.ReadAllText(<"Path to json file">));
Traverse(token, rooty);

完成了,哇塞你做到了这个: 哦不,我不允许嵌入图片。难过。


0

我用一个“根”节点包装了数据,并将其全部作为数组值进行了包装。然后这对我起作用:

    private static void TraverseJson(string jsonText)
    {
        dynamic jsonObject = JsonConvert.DeserializeObject(jsonText);

        JArray ja = (JArray)jsonObject.root;
        int itemCount = ja.Count;

        foreach (JObject jobj in jsonObject.root)
        {
            WalkHierarchy(jobj);
        }
    }

    private static void WalkHierarchy(JObject jobj)
    {
        foreach (JProperty jprop in (JToken)jobj)
        {
            if (jprop.Value.Type == JTokenType.Object)
            {
                WalkHierarchy((JObject)jprop.Value);
            }
            else
            {
                Console.WriteLine(jprop.Value.ToString());
            }
        }
    }

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