fastjson源码解析——数组反序列化总结

2021SC@SDUSC

写在前面

继上一篇fastjson源码解析——对象反序列化总结总结了fastjson的parseObject()API的大致执行顺序,本文将使用相似的方法,将fastjson的parseArray()API的架构规整为流程图、类图等描述。在这之中,parseArray使用到parseObject的部分相同方法(比如对一段token的遍历,对象的构造等对单个对象的处理),在这里不再赘叙。
本文开始综合整理parseArray的架构

正式开始

parseArray()的API有两个,一个是针对单一类型的JSON对象(parseArray(String, Class<?>)),另一个是给每一个JSON对象都规定了元素类型的parseArray(String, Type[])。虽然这是两个方法,它们的执行逻辑其实并没有很大的差距,只是在细微的地方存在差别,比如对token的处理位置,但是每个步骤进行的操作逻辑基本相同,本文不再将其分开,只是提取出共同点后稍加说明

执行顺序

1. parseArray的操作主线逻辑

数组反序列化的主线操作逻辑在两种API中并无区别,只在多级API嵌套的过程中保存了不同的类型数据:parseArray(String, Class<?>保存的是单个类型的配置信息,只向通用反序列化器中传递了1个类型配置‘parseArray(String,Type[])保存了一个类型数组Type[],使用DefaultJSONParser时传入了这个数组。
主要逻辑:

多级API嵌套
parseArray API
包含了所有配置的API
构造通用的反序列化器
检查token合法性
使用通用反序列化器反序列化
根据配置调整对象属性值
返回List类型数组
2. 构造通用的反序列化器

这一步的逻辑在不同API中并没有什么区别,只是保存了一些需要的数据,同时将token的读取位置移动到一个JSON对象或数组的开始

保存lexer
将token位置移动到下一个有效对象开始
跳转到下一位token
返回
保存输入JSON字符串
保存config配置
3. 检查token合法性

合法性检查本来很简单,只是为了检查传入的字符串是否存在不合法的JSON格式,但是其在两个API中出现了两个处理的位置
流程图:

Class<?>
Type数组
检查token是否为空
调用通用反序列化器
检查token是否存在EOF
检查token是否都是空格
在JSON.parseArray中直接操作
在parser.parseArray中检查
4. 调用DefaultJSONParser.parseArray()反序列化

两个API差别不大,更多是在细节维护程序的可读性、规范性
流程:

多层级调用
结束
parser.parseArray
包含field约束的parseArray
细化token检查
判断每个field的类型
对每个field开始反序列化
调用类似parseObject的方法
保存到List
跳转到下一个对象的token
Object数组转化为List
返回List

最后

到这里,我们JSON对象数组的反序列化的全部操作就已经完成了,我们获得了一个保存了所有对象的List数组。回顾这段历程,我们从最初的入口开始,不知不觉已经深入fastjson反序列化的底层实现,将其核心代码一一分析。我们不仅学到了开发者严密的逻辑,也看到了各种代码技巧。
本文使用合适的工具,将fastjson中parseArray()的两个API的调用逻辑、反序列化的操作顺序等简单明了地总结概述。到这里,fastjson反序列化源码分析就全部结束了,谢谢大家的关注!
对fastjson源码的解析,让我不仅仅学到了JSON字符串反序列化的操作逻辑,更让我清清楚楚地感受到阿里巴巴开发者的代码编写规范、命名规范,以及更重要的,对代码冗余精妙的把握,对高内聚、低耦合程序结构的设计等等,也让我积累了不少看代码分析代码的能力,在这里我也向fastjson开源开发团队致以我个人诚挚的敬意与谢意,没有开源社区的同行用爱发电,一次次贡献出自己的休息时间,我们也无法用到JSON解析benchmark霸榜的fastjson包,也无法看到专业开发者对代码的神级优化,谢谢!、
对fastjson反序列化源码解析的文章到此为止,谢谢大家的一路陪伴!
感谢各位老师的阅读与指导!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
fastjson 2.0.0 中,实现自定义反序列化器需要实现AutoTypeSupport接口,并在ParserConfig中注册该反序列化器。 下面是一个示例,假设我们有一个Person类: ```java public class Person { private String name; private int age; public Person() { } public Person(String name, int age) { this.name = name; this.age = age; } // getter、setter方法省略 } ``` 我们想要将一个JSON字符串反序列化成Person对象,其中JSON字符串中包含了一个自定义的type字段,用于指定反序列化成哪个类。例如: ```json { "type": "student", "name": "张三", "age": 20 } ``` 我们可以定义一个自定义的反序列化器,用于根据type字段的值反序列化成不同的类。示例代码如下: ```java public class PersonDeserializer implements ObjectDeserializer, AutoTypeSupport { @Override public <T> T deserialze(DefaultJSONParser parser, Type type, Object fieldName) { JSONObject jsonObject = parser.parseObject(); String typeValue = jsonObject.getString("type"); Class<? extends Person> clazz = getTypeClass(typeValue); if (clazz == null) { throw new JSONException("Unsupported type: " + typeValue); } return parser.parseObject(jsonObject.toJSONString(), clazz); } @Override public int getFastMatchToken() { return JSONToken.LBRACE; } @Override public boolean isAutoType(Class<?> clazz, String typeName) { return clazz == Student.class || clazz == Teacher.class; } @Override public Type resolveType(String typeName) { if ("student".equals(typeName)) { return Student.class; } if ("teacher".equals(typeName)) { return Teacher.class; } return null; } private Class<? extends Person> getTypeClass(String typeName) { Type type = resolveType(typeName); if (type instanceof Class) { Class<?> clazz = (Class<?>) type; if (Person.class.isAssignableFrom(clazz)) { return (Class<? extends Person>) clazz; } } return null; } } ``` 在上面的代码中,我们实现了ObjectDeserializer和AutoTypeSupport两个接口。其中,ObjectDeserializer接口用于实现自定义反序列化逻辑,AutoTypeSupport接口用于实现自定义的类型解析规则。 在deserialze方法中,我们首先解析JSON对象,然后根据type字段的值获取需要反序列化成的类,最后调用JSONParser的parseObject方法将JSON对象反序列化成对应的类。 在isAutoType方法中,我们指定了支持反序列化成Student和Teacher两个类。在resolveType方法中,我们根据typeName解析出对应的Class对象。 最后,在ParserConfig中注册该反序列化器: ```java ParserConfig parserConfig = new ParserConfig(); parserConfig.setAutoTypeSupport(true); parserConfig.putDeserializer(Person.class, new PersonDeserializer()); ``` 以上就是使用 fastjson 2.0.0 实现自定义反序列化器的步骤和示例代码。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值