Fastjson2中自定义Map键类型的序列化与反序列化实践
在JSON数据处理过程中,我们经常会遇到需要自定义序列化和反序列化逻辑的场景。本文将以Fastjson2项目为例,详细介绍如何处理Map中键类型转换的问题,特别是当JSON中的键为数字而我们需要将其映射为Java中的Integer类型时。
问题背景
在实际开发中,我们可能会遇到如下JSON数据结构:
{
"splitScreenDetail": {
0: ["037a00020054199eeebf", "037a00020054010a67ff"],
1: ["034a0002006e9d17f3ff"]
}
}
在Java中,我们希望将这个结构映射为Map<Integer, List<String>>类型。然而,默认情况下,JSON解析器会将对象键视为字符串类型,这就导致了类型不匹配的问题。
Fastjson1的解决方案
在Fastjson1中,可以通过实现ObjectDeserializer和ObjectSerializer接口来自定义序列化和反序列化逻辑:
@JSONField(deserializeUsing = SplitScreenDetailDeserializer.class,
serializeUsing = SplitScreenDetailDeserializer.class)
private Map<Integer, List<String>> splitScreenDetail;
public static class SplitScreenDetailDeserializer
implements ObjectDeserializer, ObjectSerializer {
@Override
public Map<Integer, List<String>> deserialze(DefaultJSONParser parser,
Type type, Object fieldName) {
Map<Object, List<String>> stringMap = parser.parseObject(HashMap.class);
Map<Integer, List<String>> resultMap = new HashMap<>();
stringMap.forEach((k,v) ->
resultMap.put(Integer.valueOf(k.toString()), v));
return resultMap;
}
@Override
public void write(JSONSerializer jsonSerializer, Object object,
Object o1, Type type, int i) throws IOException {
if (object instanceof Map) {
Map<Integer, List<String>> map = (Map<Integer, List<String>>) object;
Map<Object, List<String>> resultMap = new HashMap<>();
map.forEach((k,v) -> resultMap.put(k.toString(), v));
jsonSerializer.write(resultMap);
}
}
}
Fastjson2的改进方案
升级到Fastjson2后,API发生了变化,需要采用新的接口来实现相同的功能。Fastjson2提供了更清晰的责任分离,将序列化和反序列化逻辑分别放在不同的接口中:
反序列化实现
public static class SplitScreenDetailDeserializer
implements ObjectReader<Map<Integer, List<String>>> {
@Override
public Map<Integer, List<String>> readObject(JSONReader jsonReader,
Type fieldType,
Object fieldName,
long features) {
Map<String, JSONArray> stringMap = (Map) jsonReader.readObject();
Map<Integer, List<String>> resultMap = new HashMap<>();
stringMap.forEach((key, value) ->
resultMap.put(Integer.parseInt(key),
value.toJavaList(String.class)));
return resultMap;
}
}
序列化实现
public static class SplitScreenDetailSerializer
implements ObjectWriter<Map<Integer, List<String>>> {
@Override
public void write(JSONWriter jsonWriter, Object object,
Object fieldName, Type fieldType, long features) {
Map<String, List<String>> resultMap = new HashMap<>();
Map<Integer, List<String>> reqMap = (Map) object;
reqMap.forEach((key, value) -> resultMap.put(key.toString(), value));
jsonWriter.write(resultMap);
}
}
使用方式
定义好自定义的序列化和反序列化器后,可以通过注解方式应用到字段上:
@JSONField(deserializeUsing = SplitScreenDetailDeserializer.class,
serializeUsing = SplitScreenDetailSerializer.class)
private Map<Integer, List<String>> splitScreenDetail;
技术要点解析
-
类型安全:Fastjson2通过泛型明确了序列化和反序列化的目标类型,提高了代码的类型安全性。
-
职责分离:将序列化和反序列化逻辑分离到不同的接口中,符合单一职责原则。
-
性能优化:Fastjson2的API设计更加高效,减少了不必要的类型转换和对象创建。
-
错误处理:在实际应用中,应该增加对数字格式错误的处理,例如捕获
NumberFormatException。
最佳实践建议
-
版本兼容性:注意不同版本Fastjson的API变化,升级时需要进行充分测试。
-
性能考虑:对于高频调用的序列化/反序列化操作,可以考虑缓存转换结果。
-
日志记录:在自定义逻辑中加入适当的日志记录,便于调试和问题排查。
-
单元测试:为自定义序列化器编写全面的单元测试,覆盖各种边界情况。
通过本文的介绍,开发者可以掌握在Fastjson2中处理复杂类型转换的技巧,特别是Map键类型的自定义处理,这对于处理各种非标准JSON数据结构非常有帮助。
GLM-5智谱 AI 正式发布 GLM-5,旨在应对复杂系统工程和长时域智能体任务。Jinja00
GLM-5.1GLM-5.1是智谱迄今最智能的旗舰模型,也是目前全球最强的开源模型。GLM-5.1大大提高了代码能力,在完成长程任务方面提升尤为显著。和此前分钟级交互的模型不同,它能够在一次任务中独立、持续工作超过8小时,期间自主规划、执行、自我进化,最终交付完整的工程级成果。Jinja00
LongCat-AudioDiT-1BLongCat-AudioDiT 是一款基于扩散模型的文本转语音(TTS)模型,代表了当前该领域的最高水平(SOTA),它直接在波形潜空间中进行操作。00- QQwen3.5-397B-A17BQwen3.5 实现了重大飞跃,整合了多模态学习、架构效率、强化学习规模以及全球可访问性等方面的突破性进展,旨在为开发者和企业赋予前所未有的能力与效率。Jinja00
HY-Embodied-0.5这是一套专为现实世界具身智能打造的基础模型。该系列模型采用创新的混合Transformer(Mixture-of-Transformers, MoT) 架构,通过潜在令牌实现模态特异性计算,显著提升了细粒度感知能力。Jinja00
FreeSql功能强大的对象关系映射(O/RM)组件,支持 .NET Core 2.1+、.NET Framework 4.0+、Xamarin 以及 AOT。C#00