本篇文章已授权微信公众号 dasu_Android(大苏)独家发布

最近在 Json 数据的解析上碰到了一些坑,特此记录一下。

正文

迭代开发中,经常出现服务端接口还没开发完成的情况,所以经常需要移动端自己在本地造一些假数据。

emmm,虽然说好像造假数据也不是什么很难的事,但问题是,我是做 Tv app 的,手机 app 首页的 json 数据结构怎么样我不清楚,但 Tv 应用的主页复杂的要命,服务端下发的 json 数据格式是一层嵌套一层,每次看接口文档都一脸懵逼,接触了半年多了,我甚至对这个 json 数据结构还不是很熟悉,哎~~

举个例子吧:

咦,这么一简化,好像感觉也不是很复杂。哎,反正,实际上,整个 json 数据结构特别复杂,每一层里字段就特别多,然后还不断的嵌套。不管了,不管了,这个不是今天的主题,只是顺便抱怨一下而已。

不说废话了,回到今天的主题,注意看上图中我标箭号的地方,先提个问题:

{
"aaa":{...},
"bbb":"{...}"
}

Q1:你们觉得上面的 aaa 字段和 bbb 字段有区别么?

emmm,大伙不要鄙视我问这么基础的问题,慢慢看下去,你们就清楚我本篇想讲的是什么了。

首先,先确定下这个答案,aaa 对应的是一个新的 json 结构对象,如果要建模的话,要么直接使用 Object 对象,要么就是根据 {...} 里的结构创建一个对应的实体类;而 bbb 对应的就是一个字符串,不管 {...} 里的结构怎么样,解析的时候它就是一个 String 对象。所以,我们建模时的实体类应该就是这样吧:

public class WoZuiShuai {
private Object aaa;
private String bbb; ...
}

没错吧,那么,接下去该是造假数据了,我们填充一些值进去:

{
"aaa":{"ccc":"nifangpi"},
"bbb":"{"ddd":"wojiufangpi"}"
}

这样填充没问题吧,然后为了方便,我们不在文件里造假数据,把这个 json 数据复制到代码中:

public static String JSON = "{\n" +
" \"aaa\":{\"ccc\":\"nifangpi\"},\n" +
" \"bbb\":\"{wojiufangpi}\"\n" +
"}";

我们只需要先打个 " ",然后在这之间粘贴刚才复制的 json 串,as 会自动将转义符、换行符添加上去,没错吧,那么第二个问题来了:

Q2:你们觉得直接拿这个 JSON 数据去解析,可以得到结果么?

禁止逆向思维,不要说什么,如果能得到结果我就不会写了。严肃点,好好想想。我们可以简单的写个单元测试,测一下:

跑一下,看一下结果:

果然出错了,bbb 解析失败,那么,想明白为什么会出错了么?

哎,其实,还是自己对 json 不够了解,如果对 json 格式比较熟悉的话,一眼就看出在哪里出错了。

其实,在我们填充数据的那个步骤就已经错了。

{
"aaa":{"ccc":"nifangpi"},
"bbb":"{"ddd":"wojiufangpi"}"
}

这个 json 数据是错误的,拿到网上验证一下就清楚了,我比较习惯用 chrome 的 JSON-handle 插件,

这其实就是涉及到 json 结构如果是多层嵌套的话,内层的 " 冒号必须用转义符标志,这样计算机才能区分这个 " 是跟外层的匹配,还是跟内层的匹配。

也就是说,下面这样改完后才是正确的:

{
"aaa":{"ccc":"nifangpi"},
"bbb":"{\"ddd\":\"wojiufangpi\"}"
}

那么,对应的复制到代码里的样子是这样的:

public static String JSON = "{\n" +
" \"aaa\":{\"ccc\":\"nifangpi\"},\n" +
" \"bbb\":\"{\\\"ddd\\\":\\\"wojiufangpi\\\"}\"\n" +
"}";

这样改完后,再跑下单元测试就发现是正确的了,这个就是最近碰到的一个坑了,现在来反省下自己为什么会跳进这个坑。

反省

  1. 对 json 格式不够理解

  2. 当初是有想过转义符的问题,但看到 as 已经自动添加了转义符了,就想当然的以为转义符没问题了,其实内嵌的 " 号问题, java 本身就需要一层转义符,然后 json 也需要一层转义符,所以总的来说是需要有两层转义符,就像上图的代码块。

  3. 然后,服务端也得背点锅,因为你们给我的示例数据里就是没有转义符的,我当然以为你们是对的!!!

  4. 最后,自己造完数据其实也还是有拿去校验一遍的,但当初没注意看错误提示,插件定位到 bbb 那行结构是错的,然后就想当然的以为是 "{...}" 这外面那两个冒号的问题,想当然的以为这个冒号是多余的,就去掉了。然后更要命的是,去掉了之后的结构刚刚好是正确的,插件可以解析出来。然后拿到代码里测试时,却发现又解析不了,因为 bbb 定义的是 String 类型,但现在已经是一个 Object 类型了。所以,我的大脑就这样进入死锁了,加上冒号,插件验证格式错误,测试也通不过,去掉冒号,插件验证格式正确,但测试却还是通不过。哎,在这里卡了好久的。

以上,仅记录下来,提醒自己不要再犯傻了~~~




最近刚开通了公众号,想激励自己坚持写作下去,初期主要分享原创的Android或Android-Tv方面的小知识,感兴趣的可以点一波关注,谢谢支持~~

移动端造json假数据时的坑(转义符问题)的更多相关文章

  1. json返回数据时提示字符串超出长度

    JavaScriptSerializer json = new JavaScriptSerializer(); json.MaxJsonLength = Int32.MaxValue; return ...

  2. python 大量使用json 存储数据时,格式化输出的方式

    import json, pprint dic = {'name': 234, 'user_name': 'yan xia ting yu ', 'list': ['ds', 'a', 2], '你好 ...

  3. Android为TV端助力:RecyclerView更新数据时焦点丢失

    1.adapter的setHasStableIds设置成true 2.重写adapter的getItemId方法 @Override public long getItemId(int positio ...

  4. iOS解析Server端返回JSON数据

    在做quhao APP架构时,后台Server端使用了Java,提供WebService,而iOS和Android作为移动客户端.在做数据交互时,Server端返回JSON格式数据.由于iOS SDK ...

  5. SSM框架接收处理安卓端的json数据

    最近项目上与安卓端做JSON数据交互,使用的SSM框架,刚开始的时候感觉很简单,想着不就是把安卓端的JSON数据封装为Bean类对象吗? 于是就这样写了 可是这样一直报400,百度原因是因为请求url ...

  6. ajax 发送json数据时为什么需要设置contentType: "application/json”

    1. ajax发送json数据时设置contentType: "application/json”和不设置时到底有什么区别? contentType: "application/j ...

  7. ajax发送json数据时为什么需要设置contentType: "application/json”

    1. ajax发送json数据时设置contentType: "application/json”和不设置时到底有什么区别?contentType: "application/js ...

  8. redis存json数据时选择string还是hash

    redis存json数据时选择string还是hash 我们在缓存json数据到redis时经常会面临是选择string类型还是选择hash类型去存储.接下来我从占用空间和IO两方面来分析这两种类型的 ...

  9. [转]javascript eval函数解析json数据时为什加上圆括号eval("("+data+")")

    javascript eval函数解析json数据时为什么 加上圆括号?为什么要 eval这里要添加 “("("+data+")");//”呢?   原因在于: ...

随机推荐

  1. laravel框架基础知识点

    一.数据库:DB    1.db查    DB::table('msg')->where('id','>',$id)->get()       查询单行    DB::table(' ...

  2. 浅谈扩展欧几里得算法(exgcd)

    在讲解扩展欧几里得之前我们先回顾下辗转相除法: \(gcd(a,b)=gcd(b,a\%b)\)当a%b==0的时候b即为所求最大公约数 好了切入正题: 简单地来说exgcd函数求解的是\(ax+by ...

  3. Tomcat输出保存JVM GC日志文件

    当系统出现问题时,分析java虚拟机GC日志可以帮助我们定位问题,一般来说, 我们可以通过制定JVM参数使tomcat保存GC日志文件,具体实现如下: Windows下: 找到tomcat的解压目录, ...

  4. mysql那些事

    ---恢复内容开始--- 登录 mysql登录 -u+用户 -p 密码 显示数据库 show databases; 使用某个数据库 use xxx; 显示数据库表 show tables 显示表结构 ...

  5. git一键部署代码到远程服务器(linux)(采坑总结)

    原来一直使用FileZilla来代码部署,去年使用git,代码版本管理,真TM好用,一起回顾下历程! 一. 代码部署方式及思路: 1. 使用FTP/SFTP工具,上传代码 2. git人工部署.1. ...

  6. DxPackNet 3.音频捕捉(录音)

    用DxpackNet捕捉音频其实很简单 1.初始化控件 IDxMicrophCapture microphone; private void Form1_Load(object sender, Eve ...

  7. 针对Student表的DAO设计实例

    完整代码以及junit,mysql--connector包下载地址 : https://github.com/CasterWx/MyStudentDao 表信息: 代码: dao包----impl包- ...

  8. hdu1061(2015-N1):1.快速幂;2.找规律

    1.快速幂 原理:求a的b次方,将b转化为二进制数,该二进制位第i位的权是2^(i-1), 例如 11的二进制是1011 11 = 2³×1 + 2²×0 + 2¹×1 + 2º×1 因此,我们将a¹ ...

  9. JAVA在不确定具体 Annotation 类型时,获得注解参数

       package com.lzw.demo; @SpringBootApplication public class DemoApplication { public static void ma ...

  10. Luogu P1522 牛的旅行 Cow Tours

    题目描述 农民 John的农场里有很多牧区.有的路径连接一些特定的牧区.一片所有连通的牧区称为一个牧场.但是就目前而言,你能看到至少有两个牧区通过任何路径都不连通.这样,Farmer John就有多个 ...