api返回的json(api接口json数据)

内容广告上

00-1010

一:背景

前段时间在和一家公司调试api接口的时候,发现了一个很奇妙的问题。其api返回的json将动态变化,简化如下:

{“代码”3360101,“项目”: [{“订单标题”:“订单1”}]}

{ '代码' 3360102,'物品' : [{ '产品名称' : '商品1'}]}

逻辑是这样的:物品的内容会随着代码的变化而变化,可以是订单清单,也可以是商品清单。习惯弱类型的朋友看到这个json很正常,但是对于我们强类型的人来说,这简直就是一个绝妙的资本。你想让我反序列化什么强类型?如果你还没看懂,请看下图!

沟通后对方真的用了弱式php,花了很长时间才说服对方改变返回结构,让其直接与固有类匹配。

1. 讲故事

从商业角度来说,最好是说服对方让步,但从技术上来说,这种场景有什么好的解决方案?问题的本质是json是动态的,反序列化时不能指定匹配的类。

00-1010既然是动态的,C#中也有动态类型。为什么不用它作为json动态变化部分的接受值,把项定义为动态的。下图:

从图中:rsp。列表项返回空值,尝试失败。虽然转换失败了,相信你也看到了Newtonsoft.Json.linq.JArray,这个东西好像可以用linq操作。是的,这是临朐json。

二:寻找解决办法

有了linq foundation,提取JArray中的内容并不难。接下来,将代码更改为以下内容:

静态void Main(字符串[]参数)

{

Varjson=' { \ ' code \ ' :101,\ ' items ' 3360[{ \ ' order title ' : ' order 1 \ ' }]} ';

var rsp=JsonConvert。反序列化对象响应(JSON);

if (rsp。代码==101)

{

var items=(rsp。像JArray这样的项目)。选择(m=m['OrderTitle']。Valuestring())。to list();

控制台。WriteLine(字符串。Join(',',items));

}

if (rsp。代码==102)

{

//todo.

}

}

从代码中可以看到,我通过不同的代码做了不同的业务逻辑处理。看起来问题是通过这个半自动模型实现的,但是好奇心强的你就挖不下来吗?

00-1010

1. 使用 dynamic

我认为linq to json最大的优势在于它绕过了强类型化的限制,可以像弱类型语言一样处理、生成和读取json,这给了我们在业务处理上更多的选择。接下来,我会给你一条关于创建和查询的建议。

2. 使用 linq to json

如何构建没有强类型的json结构?顺便问一下,我想知道您是否熟悉linq to xml?记住它是如何一步一步建成的。如果你还记得,这里也有类似的建造方式,比如刚才的JArray。

JObject json = new JObject( new JProperty("Code", 101), new JProperty("Items", new JArray(new JObject() { new JProperty("OrderTitle","订单1"), new JProperty("Created",DateTime.Now) })) ); Console.WriteLine(json.ToString());

从图中看这种手工构建json的方式还是比较繁琐的,走的就是 linq to xml 的路子,有没有更简单的方式呢? 我觉得这里你可以用 C# 中的一个语法糖:匿名类型,虽然从 IL 上看也是强类型,但在用在这里太合适了,接下来我来改造一下:

JObject json = JObject.FromObject(new { Code = 101, Items = (new[] { new { OrderTitle="订单1",Created=DateTime.Now } }).ToList() }); Console.WriteLine(json.ToString());

这样是不是太方便了,算是巧用 匿名类型 吧。

2. 解析json

为了让结果更可观,我准备生成一个稍微复杂一点的json,然后通过 linq to json 和 jsonpath 两种方式操控json。

{ "store":{ "book":[ { "category":"reference", "author":"Nigel Rees", "title":"Sayings of the Century", "price":8.95 }, { "category":"fiction", "author":"Evelyn Waugh", "title":"tmdmj of Honour", "price":12.99 }, { "category":"fiction", "author":"Herman Melville", "title":"Moby Dick", "isbn":"0-553-21311-3", "price":8.99 }, { "category":"fiction", "author":"J. R. R. Tolkien", "title":"The Lord of the Rings", "isbn":"0-395-19395-8", "price":22.99 } ], "bicycle":{ "color":"red", "price":19.95 } } } 对 category 进行分组,统计每个类别的总金额 static void Main(string[] args) { var json = System.IO.File.ReadAllText("1.txt"); JObject obj = JObject.Parse(json); var dict = obj["store"]["book"].GroupBy(m => m["category"]) .ToDictionary(k => k.Key, v => v.Select(n => n.Value<decimal>("price")).Sum()); foreach (var key in dict.Keys) { Console.WriteLine(#34;key={key},value={dict[key]}"); } }

哈哈,分组统计在强大的linq面前就是这么简单!

使用 jsonpath 处理

jsonpath 就像 xmlpath 一样,非常强大,更多的功能可以参考这个网页: https://goessner.net/articles/JsonPath/。

根据上面的语法,我尝试着提取所有的price,使用 $..price 试试。

var json = System.IO.File.ReadAllText("1.txt"); JObject obj = JObject.Parse(json); var priceList= obj.SelectTokens("$..price"); foreach (var price in priceList) { Console.WriteLine(price.Value<decimal>()); }

四: 总结

我相信大家在90%的情况都是用强类型作为json的mapping,剩下的10%情况,可以了解下强大的 linq to json哈,太实用啦! 希望本篇对您有帮助。

转自 https://www.cnblogs.com/huangxincheng/p/13424556.html

内容广告下