01、前世今生
我是 fastjson,是个地地道道的杭州土著,但我始终怀揣着一颗走向全世界的雄心。这不,我在 GitHub 上的简介都换成了英文,国际范十足吧?
如果你的英语功底没有我家老板 666 的话,我可以简单地翻译下(说人话,不装逼)。
我是阿里巴巴开源的一款 JSON 解析库,可以将 Java 对象序列化成 JSON 字符串,同时也可以将 JSON 字符串反序列化为 Java 对象。
我提供了服务器端和安卓客户端两种解析工具,性能表现还不错。
我提供了便捷的方式来进行 Java 对象和 JSON 之间的互转,toJSONString()方法用来序列化,parseObject()方法用来反序列化。
我允许转换预先存在的无法修改的对象(只有 class、没有源代码)。
对 Java 泛型有着广泛的支持。
我支持任意复杂的对象(深度的继承层次)。
2012 年的时候,我就被开源中国评选为最受欢迎的国产开源软件之一。时隔多年,我的流行趋势没有丝毫减退,在 JSON 领域,我敢说我是 NO 1,因为我在 GitHub 上的粉丝数已经超过了 22k,没有任何人敢忽视我这样的成就。
02、使用指南
在使用我的 API 之前,需要先在 pom.xml 文件中引入我的依赖。
我来写一个简单的测试用例,你看一下。
Writer 是一个普通的 Java 类,有两个字段,分别是 age 和 name,还有它们俩对应的 getter 和 setter 方法。
main()方法中创建了一个 Writer 对象,然后调用我提供的一个静态方法JSON.toJSONString()来得到 JSON 字符串。
来看一下打印后的结果。
如果想反序列化的话,执行以下的代码即可。
调用静态方法 JSON.parseObject(),传递两个参数,一个是 JSON 字符串,一个是对象的类型。
如果想把 JSON 字符串转成集合的话,需要调用另外一个静态方法 JSON.parseArray()。
如果没有特殊要求的话,我敢这么说,以上 3 个方法就可以覆盖到你绝大多数的业务场景了。
03、使用注解
有时候,你的 JSON 字符串中的 key 可能与 Java 对象中的字段不匹配,比如大小写;有时候,你需要指定一些字段序列化但不反序列化;有时候,你需要日期字段显示成指定的格式。
这些特殊场景,我统统为你考虑到了,只需要在对应的字段上加上@JSONField注解就可以了。
先来看一下@JSONField注解的定义吧。
name 用来指定字段的名称,format 用来指定日期格式,serialize 和 deserialize 用来指定是否序列化和反序列化。
我建议在 getter 字段上使用 @JSONField 注解。来看一下测试代码。
此时的输出结果如下所示。
JSON 字符串中的 Age 首字母为大写,birthday 的格式符合“年月日”的预期,name 字段没有出现在结果中,说明没有被序列化。
04、序列化特性
为了满足更多个性化的需求,我在 SerializerFeature 类中定义了很多特性,你可以在调用toJSONString()方法的时候进行指定。
PrettyFormat,让 JSON 格式打印得更漂亮一些
WriteClassName,输出类名
UseSingleQuotes,key 使用单引号
WriteNullListAsEmpty,List 为空则输出 []
WriteNullStringAsEmpty,String 为空则输出“”
等等等等,更多新技能,等待你去开锁。我这里写个简单的 demo 供你参考。
对比一下配置前和配置后的结果。
05、我为什么快
众所周知,把 Java 对象序列化成 JSON 字符串,是不可能使用字符串直接拼接的,因为这样性能很差。比字符串拼接更好的办法就是使用StringBuilder。
StringBuilder 尽管已经很好了,但在性能上还有上升的空间。“自己动手,丰衣足食”,于是我就创造了一个 SerializeWriter 类,专门用来序列化。
SerializeWriter 类中包含了一个char[] buf,每序列化一次,都要做一次分配,但我使用了 ThreadLocal 来进行优化,这样就能够有效地减少对象的分配和垃圾回收,从而提升性能。
除此之外,还有很多其他的细节,比如说使用 IdentityHashMap 而不是 HashMap,既可以避免多余的 equals 操作,又可以避免多线程并发情况下的死循环。
再比如说,使用 asm 技术来避免反射导致的开销。
我承认,快的同时,也带来了一些安全性的问题。尤其是 AutoType 的引入,让黑客有了可乘之机。
在于黑客的反复较量中,我虽然变得越来越稳重成熟了,但与此同时,让我的用户为此也付出了沉重的代价。
网络上也出现了很多不和谐的声音,他们声称我是最垃圾的国产开源软件之一,只不过凭借着一些投机取巧赢得了国内开发者的信赖。
但更多的是,对我的不离不弃。
温少几乎凭一己之力撑起了一个被广泛使用 JSON 库,而其他库几乎都是靠一整个团队,就凭这一点,温少作为“初心不改的阿里初代开源人”,当之无愧。
出现漏洞并不可怕,可怕的是发现不了漏洞,或者说无法解决掉漏洞。
为了彻底解决 AutoType 带来的问题,在 1.2.68 版本中,我引入了 safeMode 的安全模式,无论白名单和黑名单,都不支持 AutoType,这样就可以彻底地杜绝攻击。
安全模式下,checkAutoType()方法会直接抛出异常。
06、尾声
不管前面的路还有多少艰难困苦,也不管还要面对多少风言风语,我都会砥砺前行,为了国产开源软件的蓬勃发展,我愿意做一个先驱者,也愿意做一个持久战者。
2020 年最后一篇文章了,我们 2021 年加油,点个赞,留住这最后一刻。
网友评论