Jackson多态反序列化:优雅处理包含基类与子类对象的JSON数组

Jackson多态反序列化:优雅处理包含基类与子类对象的JSON数组

本文详细阐述了如何使用Jackson库处理包含基类和其子类对象的JSON数组,并将其反序列化为基类类型的列表。通过在基类上应用@JsonTypeInfo和@JsonSubTypes注解,结合JsonTypeInfo.Id.DEDUCTION策略和defaultImpl配置,Jackson能够智能识别并实例化不同类型的对象,从而避免UnrecognizedPropertyException,实现灵活的多态数据映射。

问题背景:多态JSON数组的反序列化挑战

在实际的web服务交互中,我们经常会遇到json数组中包含多种类型对象的情况,这些对象可能共享一个共同的基类,但部分元素是其子类的实例,拥有基类所没有的额外属性。例如,一个车辆列表可能既包含普通轿车(car),也包含卡车(truck),其中truck是car的子类,并拥有如maxload、clearance等特有属性。

当尝试使用Jackson库将这样的JSON数组直接反序列化为List时,如果JSON中包含Truck的实例,Jackson会尝试将Truck特有的字段(如maxLoad)映射到Car类中。由于Car类中没有这些字段,Jackson会抛出com.fasterxml.jackson.databind.exc.UnrecognizedPropertyException异常,提示无法识别的属性。

考虑以下Java类定义和对应的JSON数据:

Java类定义:

public class Car {    private String make;    private String model;    private short year;    private String bodyStyle;    private String engineType;    private int horsepower;    // Getters and Setters    public String getMake() { return make; }    public void setMake(String make) { this.make = make; }    public String getModel() { return model; }    public void setModel(String model) { this.model = model; }    public short getYear() { return year; }    public void setYear(short year) { this.year = year; }    public String getBodyStyle() { return bodyStyle; }    public void setBodyStyle(String bodyStyle) { this.bodyStyle = bodyStyle; }    public String getEngineType() { return engineType; }    public void setEngineType(String engineType) { this.engineType = engineType; }    public int getHorsepower() { return horsepower; }    public void setHorsepower(int horsepower) { this.horsepower = horsepower; }    @Override    public String toString() {        return "Car{" +               "make='" + make + ''' +               ", model='" + model + ''' +               ", year=" + year +               ", bodyStyle='" + bodyStyle + ''' +               ", engineType='" + engineType + ''' +               ", horsepower=" + horsepower +               '}';    }}public class Truck extends Car {    private double maxLoad;    private double clearance;    // Getters and Setters    public double getMaxLoad() { return maxLoad; }    public void setMaxLoad(double maxLoad) { this.maxLoad = maxLoad; }    public double getClearance() { return clearance; }    public void setClearance(double clearance) { this.clearance = clearance; }    @Override    public String toString() {        return "Truck{" +               "make='" + getMake() + ''' +               ", model='" + getModel() + ''' +               ", year=" + getYear() +               ", bodyStyle='" + getBodyStyle() + ''' +               ", engineType='" + getEngineType() + ''' +               ", horsepower=" + getHorsepower() +               ", maxLoad=" + maxLoad +               ", clearance=" + clearance +               '}';    }}

JSON数据示例 (cars.json):

[  {      "make": "Ford",      "model": "Focus",      "year": 2018,      "engineType": "T4",      "bodyStyle": "hatchback",      "horsepower": 225  },  {      "make": "Toyota",      "model": "Prius",      "year": 2021,      "engineType": "T4",      "bodyStyle": "hatchback",      "horsepower": 121          },  {      "make": "Toyota",      "model": "RAV4",      "year": 2020,      "engineType": "V6",      "bodyStyle": "SUV",      "horsepower": 230          },  {      "make": "Toyota",      "model": "Tacoma",      "year": 2021,      "engineType": "V6",      "bodyStyle": "pickup",      "horsepower": 278,      "maxLoad": 1050,      "clearance": 9.4  },  {      "make": "Ford",      "model": "T150",      "year": 2017,      "horsepower": 450,      "bodyStyle": "pickup",      "maxLoad": 2320,      "clearance": 8.4  }   ]

直接使用mapper.readValue(src, new TypeReference<List>() {})进行反序列化,会遇到上述的UnrecognizedPropertyException。虽然将List替换为List可以避免异常,但这只会成功反序列化Truck实例,而普通Car实例则会被忽略或无法正确处理,导致数据丢失

解决方案:利用Jackson的@JsonTypeInfo和@JsonSubTypes

Jackson提供了强大的多态反序列化机制,通过在基类上添加@JsonTypeInfo和@JsonSubTypes注解,可以指导Jackson在运行时根据JSON内容的特征来推断并实例化正确的子类对象。

1. @JsonTypeInfo注解详解

该注解用于指定多态类型信息的处理方式。对于本场景,最合适的策略是JsonTypeInfo.Id.DEDUCTION。

Levity Levity

AI帮你自动化日常任务

Levity 206 查看详情 Levity use = JsonTypeInfo.Id.DEDUCTION: Jackson会尝试通过检查JSON对象中存在的字段来“推断”其实际类型。它会遍历所有已注册的子类型,如果某个子类型的所有特有字段都在JSON中出现,并且基类的字段也匹配,则认为该JSON对象是该子类型的实例。defaultImpl = Car.class: 这是一个关键配置。当Jackson无法通过DEDUCTION策略将JSON对象匹配到任何已注册的子类型时(例如,它就是一个纯粹的基类对象,不包含任何子类特有字段),它会回退到使用defaultImpl指定的类来实例化对象。在本例中,即Car.class。

2. @JsonSubTypes注解详解

该注解用于注册基类的所有已知子类型。Jackson在进行类型推断时,会参考这里注册的子类型列表。

@JsonSubTypes.Type(value = Truck.class): 注册Truck类作为Car的一个子类型。如果有多个子类型,可以添加多个@JsonSubTypes.Type条目。

带注解的Java类定义:

import com.fasterxml.jackson.annotation.JsonSubTypes;import com.fasterxml.jackson.annotation.JsonTypeInfo;// 在基类Car上添加多态注解@JsonTypeInfo(use = JsonTypeInfo.Id.DEDUCTION, defaultImpl = Car.class)@JsonSubTypes(@JsonSubTypes.Type(value = Truck.class))public class Car {    private String make;    private String model;    private short year;    private String bodyStyle;    private String engineType;    private int horsepower;    // 构造函数 (可选,Jackson通常使用默认无参构造函数)    public Car() {}    // Getters and Setters    public String getMake() { return make; }    public void setMake(String make) { this.make = make; }    public String getModel() { return model; }    public void setModel(String model) { this.model = model; }    public short getYear() { return year; }    public void setYear(short year) { this.year = year; }    public String getBodyStyle() { return bodyStyle; }    public void setBodyStyle(String bodyStyle) { this.bodyStyle = bodyStyle; }    public String getEngineType() { return engineType; }    public void setEngineType(String engineType) { this.engineType = engineType; }    public int getHorsepower() { return horsepower; }    public void setHorsepower(int horsepower) { this.horsepower = horsepower; }    @Override    public String toString() {        return "Car{" +               "make='" + make + ''' +               ", model='" + model + ''' +               ", year=" + year +               ", bodyStyle='" + bodyStyle + ''' +               ", engineType='" + engineType + ''' +               ", horsepower=" + horsepower +               '}';    }}public class Truck extends Car {    private double maxLoad;    private double clearance;    // 构造函数    public Truck() {}    // Getters and Setters    public double getMaxLoad() { return maxLoad; }    public void setMaxLoad(double maxLoad) { this.maxLoad = maxLoad; }    public double getClearance() { return clearance; }    public void setClearance(double clearance) { this.clearance = clearance; }    @Override    public String toString() {        return "Truck{" +               "make='" + getMake() + ''' +               ", model='" + getModel() + ''' +               ", year=" + getYear() +               ", bodyStyle='" + getBodyStyle() + ''' +               ", engineType='" + getEngineType() + ''' +               ", horsepower=" + getHorsepower() +               ", maxLoad=" + maxLoad +               ", clearance=" + clearance +               '}';    }}

完整示例代码

下面是使用上述带注解的类和JSON数据进行反序列化的完整测试代码:

import com.fasterxml.jackson.core.type.TypeReference;import com.fasterxml.jackson.databind.ObjectMapper;import java.io.IOException;import java.io.InputStream;import java.util.List;public class Test {    private static final ObjectMapper mapper = new ObjectMapper();    public static void main(String[] args) {        // 假设 "cars.json" 文件位于 classpath 中        InputStream src = Test.class.getClassLoader().getResourceAsStream("cars.json");        if (src == null) {            System.err.println("Error: cars.json not found in classpath.");            return;        }        try {            // 反序列化为 List,Jackson将根据注解自动处理多态            List cars = mapper.readValue(src, new TypeReference<List>() {});            System.out.println("Deserialization successful. Objects in the list:");            for (Car car : cars) {                System.out.println("  Object type: " + car.getClass().getName());                System.out.println("  Details: " + car.toString()); // 使用toString打印详细信息                // 可以进行类型检查并访问子类特有属性                if (car instanceof Truck) {                    Truck truck = (Truck) car;                    System.out.println("    (As Truck) Max Load: " + truck.getMaxLoad() + ", Clearance: " + truck.getClearance());                }                System.out.println("------------------------------------");            }        } catch (IOException e) {            e.printStackTrace();        }    }}

运行结果与解析

运行上述测试程序,你将看到如下输出(或类似输出):

Deserialization successful. Objects in the list:  Object type: com.yourpackage.Car  Details: Car{make='Ford', model='Focus', year=2018, bodyStyle='hatchback', engineType='T4', horsepower=225}------------------------------------  Object type: com.yourpackage.Car  Details: Car{make='Toyota', model='Prius', year=2021, bodyStyle='hatchback', engineType='T4', horsepower=121}------------------------------------  Object type: com.yourpackage.Car  Details: Car{make='Toyota', model='RAV4', year=2020, bodyStyle='SUV', engineType='V6', horsepower=230}------------------------------------  Object type: com.yourpackage.Truck  Details: Truck{make='Toyota', model='Tacoma', year=2021, bodyStyle='pickup', engineType='V6', horsepower=278, maxLoad=1050.0, clearance=9.4}    (As Truck) Max Load: 1050.0, Clearance: 9.4------------------------------------  Object type: com.yourpackage.Truck  Details: Truck{make='Ford', model='T150', year=2017, bodyStyle='pickup', engineType='null', horsepower=450, maxLoad=2320.0, clearance=8.4}    (As Truck) Max Load: 2320.0, Clearance: 8.4------------------------------------

从输出可以看出,Jackson成功地将JSON数组中的Car对象反序列化为Car实例,将包含maxLoad和clearance字段的JSON对象反序列化为Truck实例。这是因为:

当Jackson遇到一个JSON对象,它会检查是否包含Truck特有的字段(如maxLoad)。如果存在,它会推断该对象是一个Truck,并实例化Truck类。如果JSON对象不包含任何Truck特有的字段,Jackson无法将其推断为Truck。此时,defaultImpl = Car.class发挥作用,Jackson会回退并实例化一个Car对象。

这种机制使得我们能够在一个List中同时持有Car和Truck的实例,并且在需要时可以安全地进行向下转型来访问子类特有属性。

注意事项与最佳实践

DEDUCTION策略的适用性: JsonTypeInfo.Id.DEDUCTION最适用于子类拥有明确且不与基类或其他子类重叠的独特字段的情况。如果子类之间没有足够的字段差异,或者子类特有字段可能缺失,DEDUCTION可能无法准确推断类型。注册所有子类型: 务必通过@JsonSubTypes注解注册所有可能的子类型。如果未注册某个子类型,Jackson将无法识别它,并可能将其作为defaultImpl处理。其他类型推断策略: 如果DEDUCTION不适合你的场景,Jackson还提供了其他JsonTypeInfo.Id策略:Id.CLASS:JSON中需要包含一个特殊字段(通常是@class),其值是类的完全限定名。Id.NAME:JSON中需要包含一个特殊字段(通常是@type),其值是子类型在@JsonSubTypes中定义的逻辑名称。这些策略要求JSON数据中明确包含类型信息,提供了更精确的控制,但同时也增加了

以上就是Jackson多态反序列化:优雅处理包含基类与子类对象的JSON数组的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/741746.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月25日 15:46:01
下一篇 2025年11月25日 15:51:03

相关推荐

  • Golang如何开发基础的事件管理系统

    答案:Go语言中通过观察者模式实现事件管理系统,核心为事件总线。定义Event结构体与事件类型常量,构建包含handlers映射和读写锁的EventBus,提供Subscribe注册处理器、Publish异步触发回调。示例中用户创建事件触发邮件通知,主函数演示注册与发布流程,系统支持解耦、并发与扩展…

    2025年12月16日
    000
  • 如何在Golang中处理Web服务器日志_Golang Web服务器日志处理方法汇总

    使用标准库log可实现基础日志输出,结合文件写入和中间件记录请求信息;2. 采用zap、logrus或slog进行结构化日志,提升可读性与分析效率;3. 通过中间件统一记录请求响应详情,包括状态码、耗时等;4. 利用rotatelogs或logrotate实现日志轮转,避免磁盘占满;5. 合理配置多…

    2025年12月16日
    000
  • 如何在Golang中使用gRPC进行安全认证

    Golang中gRPC安全认证通过TLS加密和认证机制实现,需配置双向证书认证并启用客户端与服务端证书校验,结合Per-RPC Credentials传递Token,使用拦截器在服务端验证authorization头,确保通信安全。 在Golang中使用gRPC进行安全认证,核心方式是通过TLS加密…

    2025年12月16日
    000
  • Go语言文件传输安全:深度解析FTP、SFTP、SCP与FTPS

    本文深入探讨了Go语言中文件传输的安全性问题,特别关注了传统FTP(如`goftp`库)的固有风险。我们将详细分析FTP明文传输的弱点,并介绍更安全的替代方案,包括基于SSH的SFTP和SCP,以及基于SSL/TLS的FTPS。文章还将提供在Go语言中实现这些安全协议的指导和示例,旨在帮助开发者构建…

    2025年12月16日
    000
  • Go语言:高效读取文本文件并按行处理的全面指南

    本教程详细介绍了在go语言中读取文本文件并将其内容按行存储到字符串切片中的两种主要方法。我们将探讨使用`ioutil.readfile`结合`strings.split`的简洁方式,以及利用`bufio.scanner`进行高效逐行处理的策略,并提供相应的代码示例和最佳实践,帮助开发者根据文件大小和…

    2025年12月16日
    000
  • 如何在Golang中实现微服务灰度发布

    通过服务标签与路由策略实现灰度发布,具体包括:①在注册中心为服务实例添加version标签;②客户端基于请求特征进行负载均衡与过滤;③利用gRPC metadata传递灰度标识;④结合服务网格如Istio配置流量规则。 在Golang中实现微服务灰度发布,核心在于控制请求的流量分发,使新版本服务只接…

    2025年12月16日
    000
  • Go开发服务器防火墙弹窗解决方案:绑定本地回环地址

    在%ignore_a_1%开发过程中,频繁的代码修改导致开发服务器重编译并重启,进而触发防火墙反复请求网络权限。本文提供了一种简洁高效的解决方案:通过将go开发服务器绑定到本地回环地址(127.0.0.1),可以有效避免防火墙弹窗,确保开发流程的顺畅,同时不影响生产环境的部署配置。 问题背景与原因分…

    2025年12月16日
    000
  • Go语言crypto库在生产环境TLS客户端中的安全性评估

    本文深入探讨了Go语言内置`crypto`库作为TLS客户端在生产环境中的安全性。根据Go团队的官方观点,该库在发布初期未经过外部安全审查,且存在已知的侧信道攻击风险,例如RSA和部分椭圆曲线操作非恒定时间,以及潜在的Lucky13攻击。因此,在对安全性有极高要求的生产环境中,其适用性需谨慎评估。文…

    2025年12月16日
    000
  • Golang如何实现微服务调用链监控

    使用OpenTelemetry实现Go微服务调用链监控,通过集成otelhttp和otelgrpc中间件自动采集跨服务请求的Trace ID与Span数据,结合OTLP导出器将追踪信息发送至Jaeger等后端系统,实现调用链可视化。 Go语言实现微服务调用链监控,核心在于引入分布式追踪系统,通过统一…

    2025年12月16日
    000
  • Go语言中时间敏感代码的测试策略:使用接口模拟time.Now()

    在go语言中测试时间敏感代码时,最佳实践是采用基于接口的模拟(mocking)方法来控制时间流逝,而非修改系统时钟或尝试全局覆盖标准库。通过定义一个抽象的`clock`接口,并为其提供真实和测试用的实现,开发者可以精确地模拟时间行为,从而确保测试的隔离性、稳定性和可预测性,同时避免引入复杂的副作用。…

    2025年12月16日
    000
  • Golang命令行参数与用户输入处理:检测参数数量与空行输入

    本文深入探讨go语言中处理命令行参数和用户输入的常见问题及解决方案。针对命令行参数,我们将学习如何利用`os.args`检查参数数量并设置默认值,同时提及更高级的`flag`包。对于用户输入,特别是如何正确识别和处理用户键入回车键产生的空行输入,我们将对比`fmt.scanf`的局限性,并介绍使用`…

    2025年12月16日
    000
  • Go语言安全文件传输:告别FTP,拥抱SFTP、SCP与FTPS

    本文深入探讨了传统ftp协议在文件传输中的安全隐患,指出其明文认证和非加密传输易受网络嗅探攻击。针对go语言开发者,文章推荐了sftp、scp和ftps等更安全的替代方案,并提供了在go中实现安全文件传输的策略和注意事项,旨在帮助开发者构建健壮且安全的文件上传下载系统。 传统FTP的固有安全风险 在…

    2025年12月16日
    000
  • Go语言教程:利用网络监听实现程序单例检测的跨平台方法

    本文探讨了在go语言中实现程序单例锁或实例检测的跨平台方法。通过绑定一个本地tcp端口,程序的第一个实例可以成功获取该端口,而后续尝试启动的实例则会因端口已被占用而失败,从而有效地识别并阻止多余的程序副本运行。这种方法简洁高效,避免了复杂的操作系统特定api调用,适用于需要确保应用程序在同一时间只有…

    2025年12月16日
    000
  • Golang中如何将exec.Cmd的标准输出重定向到文件

    本文详细介绍了在golang中如何高效地将外部命令`exec.cmd`的标准输出(stdout)重定向并写入到指定文件。通过直接将文件句柄赋值给`cmd.stdout`字段,可以实现简洁且可靠的输出重定向。文章提供了清晰的代码示例,并强调了错误处理、资源管理等关键实践,帮助开发者避免常见陷阱,确保程…

    2025年12月16日
    000
  • 如何在Golang中使用unicode处理字符

    Go语言通过unicode和utf8包提供强大Unicode支持,字符串默认UTF-8编码,len()返回字节长度,需用utf8.RuneCountInString()获取字符数,遍历时应使用rune类型处理码点,unicode包可判断字符类别(如IsLetter、IsDigit)并进行大小写转换,…

    2025年12月16日
    000
  • Go语言中实现命令行交互式输入:等待并读取用户输入

    本文详细介绍了在go语言中如何实现命令行交互式输入,使用`bufio`包高效地读取用户在终端键入的整行内容。通过实例代码,演示了如何提示用户输入、等待用户键入并回车,然后将输入作为字节切片获取并处理,解决了类似java `scanner.nextline()`的功能需求,确保了程序能够顺利地与用户进…

    2025年12月16日
    000
  • Go语言中切片元素初始化与修改的正确姿势:理解for…range的迭代机制

    本文深入探讨go语言中对结构体切片进行迭代和元素修改时常见的陷阱。重点解释了`for…range`循环在单变量和双变量模式下对切片元素的不同处理方式,特别是当需要修改切片内部元素时,直接操作迭代变量可能导致的问题。文章提供了通过索引访问并修改切片元素的正确方法,以确保数据持久化,并纠正了…

    2025年12月16日
    000
  • Go语言中命令行参数与用户输入处理的最佳实践

    本文深入探讨了go语言中处理命令行参数和交互式用户输入的有效方法。针对命令行参数,我们介绍了如何使用`os.args`检查参数数量并安全地进行类型转换,同时提供了默认值处理机制。对于用户输入,文章解决了`fmt.scanf`无法识别空行的问题,推荐使用`bufio`包实现更健壮的整行读取,从而确保程…

    2025年12月16日
    000
  • Golang如何通过反射判断变量是否可设置

    可设置性指reflect.Value能否修改底层值,需通过reflect.ValueOf(变量地址).Elem()获取并调用CanSet()判断,仅当变量可寻址且非未导出字段时返回true,否则Set()将引发panic。 在Go语言中,通过反射判断一个变量是否可设置(可被修改),需要使用refle…

    2025年12月16日
    000
  • 提升MySQL INSERT语句可读性:SET语法实践指南

    本文旨在解决传统sql insert into … values 语句在处理大量列时可读性差的问题。我们将深入探讨mysql中更具可读性的insert into … set 语法,该语法通过直接将列名与值关联,显著提升了语句的清晰度和维护性,尤其适用于动态构建或调试sql语句…

    2025年12月16日
    000

发表回复

登录后才能评论
关注微信