假设我有以下JToken:
@"{
""data"": [
{
""company"": {
""ID"": ""12345"",
""location"": ""Some Location""
},
""name"": ""Some Name""
}
]
}";
我想将此标记传递给输出此JToken的FlattenToken函数:
@"{
""data"": [
{
""company_ID"": ""12345"",
""company_location"": ""Some Location"",
""name"": ""Some Name""
}
]}"
这样做的原因是我可以使用扁平的JToken并将其反序列化为DataTable。
但是,我迷失在混乱的JObjects,JTokens,JProperties和其他JMadness中。我在this post看到了答案,这很有帮助,但我仍然没有做对。这是我到目前为止所拥有的:
public static JToken FlattenToken(JToken token)
{
foreach (JToken topLevelItem in token["data"].Children())
{
foreach (JToken field in topLevelItem.Value<JToken>())
{
foreach (JProperty property in field.Value<JObject>().Properties())
{
field.AddAfterSelf(JObject.Parse(@"{""" + property.Name + "_" + property.Value));
}
field.Remove();
}
}
return token;
}
通过外部foreach循环的第一次迭代,topLevelItem =
{
"company": {
"ID": "12345"
},
"name": "Some Name"
}
通过第二个foreach循环的第一次迭代,field =
"company": {
"ID": "12345"
}
到目前为止看起来不错。但是当我触及最里面的foreach循环时,我在foreach行上得到一个异常:&#34;无法将Newtonsoft.Json.Linq.JProperty强制转换为Newtonsoft.Json.Linq.JToken。&#34;
不确定那里发生了什么。我的印象是field.Value调用将生成一个JToken并尝试将其转换为JProperty。那么,正如错误所暗示的那样,JProperty试图被转换为JToken的位置在哪里?
此外,这感觉就像是一种夸大JToken的非常粗暴的方式。还有更好的方法吗?
答案 0 :(得分:5)
Json.NET中的对象层次结构可能相当深。可以在this answer中找到粗略的指南。
要解决您的问题,首先需要一个扩展方法来获取JObject
的属性,然后返回一个名称前缀为的集合:
public static class JsonExtensions
{
public static IEnumerable<KeyValuePair<string, JToken>> FlattenFields(this JObject obj, string prefix)
{
foreach (var field in obj)
{
string fieldName = prefix + "_" + field.Key;
var fieldValue = field.Value;
yield return new KeyValuePair<string, JToken>(fieldName, fieldValue);
}
}
}
接下来,您需要一些递归工具来遍历Json.NET层次结构并重写所选JObject
的属性集合:
public static class JsonExtensions
{
public static IEnumerable<T> Yield<T>(this T item)
{
yield return item;
}
public static JToken EditFields(this JToken token, Func<KeyValuePair<string, JToken>, IEnumerable<KeyValuePair<string, JToken>>> editor)
{
if (token == null)
return null;
switch (token.Type)
{
case JTokenType.Array:
return EditFields((JArray)token, editor);
case JTokenType.Object:
return EditFields((JObject)token, editor);
default:
return token;
}
}
static JToken EditFields(JArray array, Func<KeyValuePair<string, JToken>, IEnumerable<KeyValuePair<string, JToken>>> editor)
{
JArray newArray = null;
foreach (var element in array)
{
var newElement = EditFields(element, editor);
if (newElement != null)
{
if (newArray == null)
newArray = new JArray();
newArray.Add(newElement);
}
}
return newArray;
}
static JToken EditFields(JObject obj, Func<KeyValuePair<string, JToken>, IEnumerable<KeyValuePair<string, JToken>>> editor)
{
JObject newObj = null;
foreach (var field in obj)
{
foreach (var newField in editor(field))
{
if (newObj == null)
newObj = new JObject();
newObj[newField.Key] = newField.Value.EditFields(editor);
}
}
return newObj;
}
}
最后,将这些放在一起以创建一个方法,将属性命名为JObject
的属性提升为其父JObject
,并在属性名称前加上下划线:
public static class JsonExtensions
{
public static JToken PromoteNamedPropertiesToParents(this JToken token, string propertyName)
{
return token.EditFields(pair =>
{
if (pair.Key == propertyName && pair.Value is JObject)
{
return ((JObject)pair.Value).FlattenFields(pair.Key);
}
return pair.Yield();
});
}
}
然后,测试:
public static class TestFlatten
{
public static void Test()
{
string jsonString = @"{
""data"": [
{
""company"": {
""ID"": ""12345"",
""location"": ""Some Location""
},
""name"": ""Some Name""
}
]
}";
JObject obj = JObject.Parse(jsonString);
var newObj = (JObject)obj.PromoteNamedPropertiesToParents("company");
Debug.WriteLine(newObj);
}
}
输出是:
{
"data": [
{
"company_ID": "12345",
"company_location": "Some Location",
"name": "Some Name"
}
]
}
这是你想要的。请注意,此代码会创建新的JObject
层次结构,而不是修改原始层次结构。