一:背景
1. 讲故事
前段时间和一家公司联调api接口的时候,发现一个奇葩的问题,它的api返回的json会动静扭转,简化如下:
{"Code":101,"Items":[{"OrderTitle":"订单1"}]}{"Code":102,"Items":[{"ProductTitle":"商品1"}]}
逻辑是这样的: Items
中的内容会随的 Code 的扭转而扭转,外面有可能是订单列表又有可能是商品列表,习惯弱类型的敌人看这种json太失常不过了,但对于强类型的咱们来说,几乎就是一个大写的奇葩,你这让我用什么强类型反序列化呢???,如果还没了解,请看上面的这张图吧!
通过沟通,对方果然用的是弱类型的php,磨了半天,压服让对方改了返回构造,这样就能够间接用固有类匹配。
二:寻找解决办法
从业务上来说,能压服对方退让那是最好的,但从技术上来说,这种场景有什么好的解决办法呢? 问题的实质就是json是动静的,你反序列化的时候无奈指定匹配类。
1. 应用 dynamic
既然是动静的,那C#中也有一个动静类型 dynamic,何不用它来做json中动态变化的那局部的承受值,将 items
定义为 dynamic。如下图:
从图中看: rsp.Items as List<OrderItem>
返回是null,尝试失败,尽管转化失败了,但我置信你也看到了 Newtonsoft.Json.Linq.JArray
,貌似这玩意能够用 linq 操控,对的, 这就是 linq to json
。
2. 应用 linq to json
有了linq根底,提取JArray中内容就不难了,接下来把代码改成如下:
static void Main(string[] args) { var json = "{\"Code\":101,\"Items\":[{\"OrderTitle\":\"订单1\"}]}"; var rsp = JsonConvert.DeserializeObject<ApiResponse>(json); if (rsp.Code == 101) { var items = (rsp.Items as JArray).Select(m => m["OrderTitle"].Value<string>()).ToList(); Console.WriteLine(string.Join(",", items)); } if (rsp.Code == 102) { //todo .... } }
从代码中能够看到,我是通过code的不同做了不同的业务逻辑解决,貌似问题通过这种半自动化的model实现了,但领有弱小好奇心的你,岂能不往下挖?
三: linq to json 剖析
1. 益处
我感觉 linq to json 的最大益处就是绕过了强类型限度,能够像弱类型语言一样解决生成和读取json,给了咱们在业务解决上更多的抉择余地,接下来我就在Create和Query上给大家抛砖引玉吧。
2. 生成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":"Sword 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($"key={key},value={dict[key]}"); } }
哈哈,分组统计在弱小的linq背后就是这么简略!
- 应用 jsonpath 解决
jsonpath 就像 xmlpath 一样,十分弱小,更多的性能能够参考这个网页: https://goessner.net/articles...。
依据下面的语法,我尝试着提取所有的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哈,太实用啦! 心愿本篇对您有帮忙。