Как сделать рекурсивный спуск JSON с помощью json.net?
Я пытаюсь проанализировать файл JSON с помощью json.net. Файл выглядит так
{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)
{
...
}
});
}
6 ответов
Код ниже должен быть очень близок к тому, что вы ищете. Я сделал предположение, что существует внешний массив, и что массивы могут появляться в любом месте иерархии. (Если это не так, вы можете немного упростить код метода 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);
}
}
}
}
}
Также необходимо сделать что-то в этом роде. Хотел бы предложить мое решение. Это имеет преимущество:
- не быть рекурсивным
- нет обратных вызовов
- не предполагая никакой внутренней структуры (массивы)
отделяет обход дерева от действия, которое необходимо выполнить
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); } } }
Затем вы можете использовать linq для фильтрации и выполнения действий:
var tokens = AllTokens(jsonObj); var titles = tokens.Where(t => t.Type == JTokenType.Property && ((JProperty)t).Name == "Title");
Я подумал, что добавлю свои незначительные изменения в метод WalkNode @BrianRogers, сделав его немного более универсальным:
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);
}
});
Вы также можете сделать это с помощью JSONPath: node.SelectTokens("$..*");
Используется так:
var jObjectsWithTitle = node
.SelectTokens("$..*")
.OfType<JObject>()
.Where(x => x.Property("Title") != null);
Или просто:
var jObjectsWithTitle = node.SelectTokens("$..[?(@.Title)]");
Попробуйте этот метод, который я написал после нескольких неудачных попыток:
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);
Готово, Бом, ты получил это: О, нет, мне не разрешено вставлять картинки. печальный.
Я заключил данные в «корневой» узел и все это как значение массива. Тогда это работает для меня:
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());
}
}
}