首页
/ Jackson-databind 中 JSON 整数反序列化为 double 类型构造参数的问题解析

Jackson-databind 中 JSON 整数反序列化为 double 类型构造参数的问题解析

2025-06-20 02:01:58作者:邬祺芯Juliet

问题背景

在 Java 开发中,Jackson 是一个非常流行的 JSON 处理库。最近在 Jackson-databind 项目中,开发者发现了一个关于 JSON 数值反序列化的有趣问题:当尝试将一个 JSON 整数(如 5)反序列化为一个带有单个 double 类型参数的构造函数时,会抛出 MismatchedInputException 异常,而同样的 JSON 数值如果写成浮点形式(如 5.0)则可以正常工作。

问题复现

考虑以下简单的 Java 类:

public static final class Stuff {
    public final double value;
    
    public Stuff(double value) {
        this.value = value;
    }
}

当尝试用以下代码进行反序列化时:

ObjectMapper mapper = new ObjectMapper();
Stuff a = mapper.readValue("5", Stuff.class);

这段代码会抛出异常,而如果将 JSON 值改为 5.0 则可以正常工作。

技术分析

当前行为分析

Jackson 当前的反序列化机制在处理单参数构造函数时,对于数值类型的匹配相对严格。当 JSON 提供整数而构造函数期望 double 类型时,默认情况下不会自动进行类型转换。

然而,有趣的是,如果使用 @JsonProperty 注解标记字段,Jackson 却能正确处理这种类型转换:

public static final class Stuff2 {
    public final double value;
    
    public Stuff2(@JsonProperty("value") double value) {
        this.value = value;
    }
}

这种不一致的行为表明,Jackson 内部对字段反序列化和构造函数参数反序列化的处理机制存在差异。

解决方案讨论

经过社区讨论,决定修改 StdValueInstantiator 类的行为,使其能够自动将 JSON 整数转换为构造函数期望的 double 类型参数。这种修改保持了与 @JsonProperty 注解行为的一致性,同时也符合大多数开发者的直觉预期。

实现细节

解决方案主要涉及修改 StdValueInstantiator 类中的以下方法:

  1. createFromInt() - 当构造函数参数为 double 类型时,将输入的 int 值转换为 double
  2. createFromLong() - 同样处理 longdouble 的转换

这种修改需要考虑数值精度问题:

  • intdouble 的转换不会丢失精度
  • longdouble 的转换可能会丢失精度,但这是开发者选择使用 double 类型时应该承担的责任

扩展讨论

在进一步讨论中,开发者还提出了对其他数值类型的支持,如 floatshortbyte。这些类型的处理需要考虑更多因素:

  1. 精度损失 - 特别是从 longfloat 的转换
  2. 数值范围 - 确保转换不会导致溢出
  3. 与现有 @JsonProperty 行为的一致性

目前决定先实现 float 的支持,其他类型将在后续评估后再决定是否添加。

总结

这个改进使得 Jackson 的反序列化行为更加一致和符合直觉,减少了开发者需要编写的样板代码。它也展示了开源社区如何通过讨论和协作来解决实际开发中遇到的问题,最终提升框架的整体用户体验。

对于开发者来说,这意味着在使用 Jackson 时,可以更自然地处理数值类型的转换,而不必为每种可能的数值类型都显式地编写构造函数或使用注解。

登录后查看全文
热门项目推荐

项目优选

收起
kernelkernel
deepin linux kernel
C
22
6
docsdocs
OpenHarmony documentation | OpenHarmony开发者文档
Dockerfile
166
2.05 K
nop-entropynop-entropy
Nop Platform 2.0是基于可逆计算理论实现的采用面向语言编程范式的新一代低代码开发平台,包含基于全新原理从零开始研发的GraphQL引擎、ORM引擎、工作流引擎、报表引擎、规则引擎、批处理引引擎等完整设计。nop-entropy是它的后端部分,采用java语言实现,可选择集成Spring框架或者Quarkus框架。中小企业可以免费商用
Java
8
0
openHiTLS-examplesopenHiTLS-examples
本仓将为广大高校开发者提供开源实践和创新开发平台,收集和展示openHiTLS示例代码及创新应用,欢迎大家投稿,让全世界看到您的精巧密码实现设计,也让更多人通过您的优秀成果,理解、喜爱上密码技术。
C
88
568
leetcodeleetcode
🔥LeetCode solutions in any programming language | 多种编程语言实现 LeetCode、《剑指 Offer(第 2 版)》、《程序员面试金典(第 6 版)》题解
Java
60
17
apintoapinto
基于golang开发的网关。具有各种插件,可以自行扩展,即插即用。此外,它可以快速帮助企业管理API服务,提高API服务的稳定性和安全性。
Go
22
0
cjoycjoy
一个高性能、可扩展、轻量、省心的仓颉应用开发框架。IoC,Rest,宏路由,Json,中间件,参数绑定与校验,文件上传下载,OAuth2,MCP......
Cangjie
94
15
ohos_react_nativeohos_react_native
React Native鸿蒙化仓库
C++
199
279
giteagitea
喝着茶写代码!最易用的自托管一站式代码托管平台,包含Git托管,代码审查,团队协作,软件包和CI/CD。
Go
17
0
RuoYi-Vue3RuoYi-Vue3
🎉 (RuoYi)官方仓库 基于SpringBoot,Spring Security,JWT,Vue3 & Vite、Element Plus 的前后端分离权限管理系统
Vue
954
564