一:背景

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/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哈,太实用啦!希望本篇对您有帮助。

api接口返回动态的json格式?我太难了,尝试一下 linq to json相关推荐

  1. 分布式之API接口返回格式如何优雅设计?,苦熬一个月

    #2000-2999 区间表示用户错误 #3000-3999 区间表示接口异常 这样前端开发人员在得到返回值后,**根据状态码就可以知道,大概什么错误**,再根据message相关的信息描述,**可以 ...

  2. 数据传输:json格式序列化与反序列化(将类转化为json格式与将json格式转化为类)

    数据传输:json格式序列化与反序列化(将类转化为json格式与将json格式转化为类) 在最近的实验中,使用了json来传送客户端与服务端的信息,之前原本想使用自己定义的格式,但是使用json可以将 ...

  3. json最大长度限制_api接口返回动态的json格式?我太难了,尝试一下 linq to json

    一:背景 1. 讲故事 前段时间和一家公司联调api接口的时候,发现一个奇葩的问题,它的api返回的json会动态改变,简化如下: {"Code":101,"Items& ...

  4. 基于“新浪乐库”的音乐搜索API接口,返回JSON数据

    新浪开发平台提供的API接口非常丰富,可没有音乐搜索的接口,对于我们这些搞基于微博应用的孩子还是比较悲剧的. 于是我可爱的好基友@ACMer_耕畅写了一段代码,把"新浪乐库"返回的 ...

  5. API接口 返回值说明

    ​介绍API:应用程序接口(API:Application Program Interface) 应用程序接口是一组定义.程序及协议的集合,通过 API 接口实现计算机软件之间的相互通信.API 的一 ...

  6. 【虾皮shopee来赞达lazada】各区域商品详情API接口返回值说明

    虾皮shopee电商数据(来赞达lazada同理) 虾皮(shopee)是一个亚洲区域的电商平台,主要在东南亚地区提供电商服务.虾皮提供了丰富的电商数据,包括商品数据.订单数据.会员数据.评价数据等. ...

  7. 淘宝/天猫搜索同款的商品 API 接口返回值说明

    前言 通过根据店铺商品来提取平台同款现有的商品或者潜在的竞争对手的产品进行比较分析.我们可以查询到市面上有没有和自己产品类似的竞品.同时快速了解到这块领域当前是处于什么阶段?蓝海?还是红海?并且快速了 ...

  8. java发送post请求json格式_Linux QT 4G发送HTTP POST请求发送JSON格式的数据

    一.在PC Window(QT5版本) 1.1.构建URL 1.2.构建HTTP请求头 1.3.连接信号与操作,接收服务器回复的消息 槽函数这里我们要在头文件定义,然后把处理过程实现. void Ma ...

  9. ajax 中json格式数据格式,AJAX中的dataType(数据格式)-text、json

    因为经常使用数据格式,所以将它封装成类,J这样就不会用到时就写了,直接调用写好的类就可以了 (1)dataType数据格式为:TEXT格式的数据是字符串的数据,在"ajax对数据进行删除和查 ...

最新文章

  1. 灵活运用 SQL SERVER FOR XML PATH
  2. ios开发学习笔记--调用相册或相机(UIImagePickerController)
  3. NOSQL 之 cassadra 安装与集群配置
  4. SQL Server 2005 COM+ 目录要求”警告
  5. html从入门到卖电脑(三)
  6. Linux 网络相关命令
  7. mdm.Dialogs.promptModal()使用
  8. SQL SERVER 2005 数据挖掘与商业智能完全解决方案---学习笔记(二)
  9. 日语口语中需要注意的问题~ 不定期更新
  10. 国二计算机考试office快捷键,全国计算机二级考试(Office)应试技巧
  11. java图书借阅系统java图书管理系统java书籍借阅系统
  12. 东芝2303am维护清零_东芝打印机2303A怎样清零
  13. http测试工具:httpbin
  14. 2013年度CSDN十大博客之星
  15. 推荐ietester工具
  16. 微信小程序——天气查询
  17. 2013全年3GPP RAN1会议关于D2D(Device-to-Device)技术的提案分析
  18. request.setattribute详解
  19. 出来找工作,去创业公司好吗?
  20. 物理层设备(中继器和集线器)

热门文章

  1. canvas特效代码详解(2)
  2. 理解T-SQL: 脚本和批处理
  3. React 深入学习:React 更新队列
  4. python应用POP3、IMAP、SMTP 协议,获取邮箱验证码
  5. 边写 Javascript 代码边玩游戏 – WarriorJS
  6. 怎样用原生js配合css的transition写个无缝滚动
  7. Linux日志出现大量kernel: NET: Registered protocol family 36
  8. Centos7 安装gitlab 8.7.5
  9. github 上微信判断是否被删除的源码 以及使用解惑
  10. 为什么PostgreSQL比MongoDB还快之完结篇(深挖单点索引查询)