一:背景

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

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

  1. api接口返回动态的json格式?我太难了,尝试一下 linq to json

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

  2. uniapp前端处理接口返回一整个html格式

    问题描述 调用了一个接口,返回的不是常规的json数据,是一个html页面的html代码 正常需要返回展示的页面是这样的 一开始想着用富文本解析做,发现view层无法解析全部的html代码,比如还有s ...

  3. react-native 展示后台获取的接口返回的html文本协议内容的几种尝试以及最佳尝试

    1.使用WebView,展示效果如下(字太小,还需要用户去放大,这种情况只在ios出现,android正常大小,因此我们需要引入一个个scalesPageToFit属性给WebView,即:scale ...

  4. Echarts获取国家json文件(非洲国家地图,动态选中地图上国家)

    前言 需求:使用echarts构建非洲地图并通过下拉框动态选中地图中得国家. 思路:网上没有现成的非洲国家地图json坐标数据,只有世界国家地图json坐标数据,所以首先找到非洲所有国家中英文名称,然 ...

  5. php接口返回一个数组怎末写_php api返回json数组

    [PHP] 为JSON数据的API返回空数组或者空对象 PHP 中,不管是 list 或者 dictionary 都使用一样的 [](或者 array()) 来定义. 在使用 JSON 作为 API ...

  6. django通过ajax请求接口返回多条数据,并动态生成表格,请求表单后将表格数据并入库

    一.最近在做接口相关的开发,需求是这样的,通过一个接口所需要传递的参数,调用接口后,处理接口响应的参数,返回多条数据,并动态生成表格,请求表单后将表格的数据入库,下面是我改过的代码,跟实际代码有些出入 ...

  7. 接口返回html转换josn,接口返回数据Json格式处理

    有这样一个页面 , 用来显示用户的账户记录数据,并且需要显示每个月的 收入 支出合计 ,在分页的时候涉及到一些问题,需要对返回的Json格式做处理,处理起来比较麻烦,后端返回的Json数据格式形式如下 ...

  8. 如何用java语法解析接口返回的json串?

    起因:做接口测试的时候经常需要对接口返回的响应数据(一般都是json字符串格式)做解析 下面是一个接口响应数据的例子: 下面是对上面接口返回的json格式字符串的解析 Object dataObj=J ...

  9. springboot 接口返回数据时 net.sf.json.JSONNull[“empty“]) 异常

    springboot 接口返回数据时 net.sf.json.JSONNull["empty"]) 异常 参考文章: (1)springboot 接口返回数据时 net.sf.js ...

最新文章

  1. 武汉.NET俱乐部论坛已经恢复
  2. 上海首次正式试用人脸识别系统抓医药代表:频繁出入的非就医可疑人员
  3. python函数中可变参数的传递方式是_Python函数可变参数定义及其参数传递方式实例详解...
  4. Redis 会遇到的「坑」,你踩过几个?
  5. python给生活带来哪些改变_Python - 笔记1
  6. Linus 发文宣布Linux Kernel 5.0 正式发布
  7. WPF系列:GridView列绑定控件(一)
  8. happy 2016, happy 11111100000
  9. timeSetEvent的用法(一)
  10. 图:杭州中级人民法院演讲比赛于9月29日圆满结束.其主题图片展示。
  11. 运行时的相互联系(PE、Windows 加载器、应用程序域、程序集清单、元数据、类型、对象、线程栈、托管堆)...
  12. ae合成复制脚本_【脚本】AE脚本精选系列 | 合成复制脚本 True Comp Duplicator v3.9.7...
  13. 聊聊六位半万用表电路(一)——保护
  14. 『深度实战』天池小目标检测大赛·宫颈癌风险智能诊断推荐
  15. java tapestry_Java Web 框架 Tapestry
  16. 帮我用js写一个微信聊天那种气泡效果
  17. 手机端富文本编辑器_在手机上也能高效写作,这款好用的移动端编辑器值得你尝试...
  18. 图文详解!10大高性能开发核心技术+
  19. python 之复数
  20. android 11中置入第三方应用apk

热门文章

  1. 一些常用的meta标签及其作用
  2. [VC]strcpy memcpy memset区别与简介
  3. (转)Kinect背景移除支持多人
  4. 使用Excel VBA(快捷键)(加菜单)
  5. Qt5:渐变效果的实现
  6. 【转】近期Coolite控件的技术点总结
  7. java处理unicode_C# JavaScript Java 与 中文 unicode 处理
  8. python实现邮件客户端_利用python实现简单的邮件发送客户端示例
  9. 快搜浏览器_让微软丢大脸的edge浏览器终于出新版了!
  10. 吴恩达《机器学习》学习笔记十四——应用机器学习的建议实现一个机器学习模型的改进