Fastjson2中JSONB序列化不可变集合的Bug分析与修复
问题背景
在Fastjson2项目中,当使用JSONB格式进行序列化和反序列化操作时,如果对象中存在默认值为不可变集合(如Collections.emptyList())的字段,并且开启了FieldBased和NotWriteHashMapArrayListClassName特性时,会出现反序列化失败的问题。
问题现象
具体表现为:当反序列化一个包含不可变集合默认值的对象时,Fastjson2会尝试向这个不可变集合中添加元素,导致抛出UnsupportedOperationException异常。这种情况常见于MyBatis-Plus的分页结果对象Page中,其records字段默认值为Collections.emptyList()。
技术分析
问题的根源在于Fastjson2的反序列化机制。在反序列化过程中,Fastjson2会先获取字段的默认值(即不可变集合),然后尝试向这个集合中添加反序列化得到的元素。然而,不可变集合(如Collections.emptyList()返回的集合)是不支持修改操作的,因此会抛出异常。
在Fastjson2的源码中,这个问题出现在ObjectReaderCreatorASM类的genReadFieldValueList方法中。该方法生成的字节码会直接使用字段的默认值,而没有考虑这个默认值是否可修改。
解决方案
Fastjson2开发团队在2.0.52版本中修复了这个问题。修复方案主要包括:
- 在反序列化集合类型字段时,首先检查默认值是否是可修改的集合
- 如果默认值是不可修改的集合,则创建一个新的可修改集合实例
- 将反序列化得到的元素添加到新创建的集合中
- 最后将新集合设置回对象字段
这种处理方式既保证了反序列化的正确性,又不会影响原有不可变集合的语义。
影响范围
该问题主要影响以下场景:
- 使用JSONB格式进行序列化和反序列化
- 开启了
FieldBased和NotWriteHashMapArrayListClassName特性 - 对象中包含默认值为不可变集合的字段
- 包括但不限于
Collections.emptyList()、Collections.emptySet()和Collections.emptyMap()
最佳实践
为了避免类似问题,开发者在使用Fastjson2时应注意:
- 对于可能被序列化的对象,谨慎使用不可变集合作为字段默认值
- 如果确实需要使用不可变集合,可以考虑在反序列化后手动处理
- 及时升级到Fastjson2的最新版本以获取bug修复
总结
Fastjson2作为高性能的JSON处理库,在处理复杂对象序列化时需要考虑各种边界情况。这个bug的修复体现了开发团队对细节的关注和对稳定性的追求。开发者在使用过程中遇到类似问题时,应及时检查版本并考虑升级到修复版本。
Kimi-K2.5Kimi K2.5 是一款开源的原生多模态智能体模型,它在 Kimi-K2-Base 的基础上,通过对约 15 万亿混合视觉和文本 tokens 进行持续预训练构建而成。该模型将视觉与语言理解、高级智能体能力、即时模式与思考模式,以及对话式与智能体范式无缝融合。Python00
PaddleOCR-VL-1.5PaddleOCR-VL-1.5 是 PaddleOCR-VL 的新一代进阶模型,在 OmniDocBench v1.5 上实现了 94.5% 的全新 state-of-the-art 准确率。 为了严格评估模型在真实物理畸变下的鲁棒性——包括扫描伪影、倾斜、扭曲、屏幕拍摄和光照变化——我们提出了 Real5-OmniDocBench 基准测试集。实验结果表明,该增强模型在新构建的基准测试集上达到了 SOTA 性能。此外,我们通过整合印章识别和文本检测识别(text spotting)任务扩展了模型的能力,同时保持 0.9B 的超紧凑 VLM 规模,具备高效率特性。Python00
xw-cli实现国产算力大模型零门槛部署,一键跑通 Qwen、GLM-4.7、Minimax-2.1、DeepSeek-OCR 等模型Go06
yuanrongopenYuanrong runtime:openYuanrong 多语言运行时提供函数分布式编程,支持 Python、Java、C++ 语言,实现类单机编程高性能分布式运行。Go051
pc-uishopTNT开源商城系统使用java语言开发,基于SpringBoot架构体系构建的一套b2b2c商城,商城是满足集平台自营和多商户入驻于一体的多商户运营服务系统。包含PC 端、手机端(H5\APP\小程序),系统架构以及实现案例中应满足和未来可能出现的业务系统进行对接。Vue00
ebook-to-mindmapepub、pdf 拆书 AI 总结TSX01