0

0

Jackson深度克隆DTO时忽略特定类型数组属性的策略

聖光之護

聖光之護

发布时间:2025-10-15 09:33:31

|

204人浏览过

|

来源于php中文网

原创

Jackson深度克隆DTO时忽略特定类型数组属性的策略

在spring boot应用中,当需要深度克隆包含非序列化字段(如`multipartfile`及其数组)的dto时,传统的`objectoutputstream`方法不可行。本文将探讨如何利用jackson的`objectmapper`和`addmixin`机制,巧妙地忽略这些不可控dto中的特定类型数组属性,从而实现dto的顺利克隆,并指出处理集合类型时的局限性。

背景与挑战

在开发过程中,我们经常需要对DTO(数据传输对象)进行深度克隆,例如在处理请求参数时。然而,当DTO中包含如org.springframework.web.multipart.MultipartFile这类非序列化接口的字段时,传统的Java序列化(ObjectOutputStream和ObjectInputStream)方法会失效。更具挑战性的是,有时我们无法修改这些DTO的定义,例如通过添加transient关键字或@JsonIgnore注解来排除字段。

Jackson库提供了一种灵活的解决方案,即通过ObjectMapper的addMixIn()方法结合@JsonIgnoreType注解,在不修改原始类定义的情况下,指示Jackson忽略特定类型的字段。

初始尝试与局限

为了忽略DTO中的MultipartFile字段,我们可能会尝试以下Jackson配置:

import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.annotation.JsonIgnoreType;
import org.springframework.web.multipart.MultipartFile;

public class DtoCloner {

    private Object makeClone(Object obj) {
        ObjectMapper mapper = new ObjectMapper();
        // 添加针对 MultipartFile 类型的 MixIn
        mapper.addMixIn(MultipartFile.class, JacksonMixInForIgnoreType.class);
        try {
            // 将对象序列化为JSON字符串,再反序列化回对象
            return mapper.readValue(mapper.writeValueAsString(obj), obj.getClass());
        } catch (JsonProcessingException e) {
            throw new RuntimeException("DTO克隆失败", e);
        }
    }

    // 定义一个空的 MixIn 接口或类,并用 @JsonIgnoreType 标记
    @JsonIgnoreType
    static class JacksonMixInForIgnoreType {}

    // 示例DTO,可能包含 MultipartFile 字段
    static class DeepCopyDto {
        private String name;
        private MultipartFile singleFile;
        // ... 其他字段
    }
}

上述代码对于DTO中包含单个MultipartFile字段的情况通常是有效的。然而,当DTO中包含MultipartFile的数组,例如MultipartFile[] fileArray;时,这种策略会失败,并抛出InvalidDefinitionException异常,指出无法找到java.io.FileDescriptor的序列化器,错误信息大致如下:

com.fasterxml.jackson.databind.exc.InvalidDefinitionException: No serializer found for class java.io.FileDescriptor ... (through reference chain: com.example.uploadingfiles.DeepCopyDto["fileArray"]->org.springframework.web.multipart.support.StandardMultipartHttpServletRequest$StandardMultipartFile[0]->org.springframework.web.multipart.support.StandardMultipartHttpServletRequest$StandardMultipartFile["inputStream"]->java.io.FileInputStream["fd"])

这表明Jackson在处理数组类型时,并未将MultipartFile[]视为与MultipartFile相同的类型进行忽略,而是尝试深入数组内部,最终遇到无法序列化的底层IO流对象。

千面数字人
千面数字人

千面 Avatar 系列:音频转换让静图随声动起来,动作模仿让动漫复刻真人动作,操作简单,满足多元创意需求。

下载

解决方案:明确指定数组类型的MixIn

问题的根源在于Jackson在应用MixIn时,将MultipartFile和MultipartFile[]视为两种不同的类型。因此,要忽略MultipartFile数组,我们需要为数组类型本身也添加一个MixIn。

修正后的makeClone方法如下:

import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.annotation.JsonIgnoreType;
import org.springframework.web.multipart.MultipartFile;

public class DtoCloner {

    private Object makeClone(Object obj) {
        ObjectMapper mapper = new ObjectMapper();
        // 1. 添加针对 MultipartFile 类型的 MixIn
        mapper.addMixIn(MultipartFile.class, JacksonMixInForIgnoreType.class);
        // 2. 额外添加针对 MultipartFile 数组类型的 MixIn
        mapper.addMixIn(MultipartFile[].class, JacksonMixInForIgnoreType.class); // 关键改动
        try {
            return mapper.readValue(mapper.writeValueAsString(obj), obj.getClass());
        } catch (JsonProcessingException e) {
            throw new RuntimeException("DTO克隆失败", e);
        }
    }

    @JsonIgnoreType
    static class JacksonMixInForIgnoreType {}

    // 示例DTO,现在可以包含 MultipartFile 数组
    static class DeepCopyDto {
        private String name;
        private MultipartFile singleFile;
        private MultipartFile[] fileArray; // 包含 MultipartFile 数组
        // ... 其他字段
    }
}

通过在ObjectMapper中额外添加一行mapper.addMixIn(MultipartFile[].class, JacksonMixInForIgnoreType.class);,我们明确告诉Jackson:当遇到MultipartFile类型或MultipartFile数组类型时,都应该应用JacksonMixInForIgnoreType,从而忽略这些字段的序列化。

注意事项与局限性

  1. 类型匹配的精确性: Jackson的addMixIn机制要求精确匹配类型。MultipartFile和MultipartFile[]在JVM中是不同的类型,因此需要分别配置。
  2. 集合类型(如List): 值得注意的是,上述方法对于泛型集合类型(如List)并不直接适用。addMixIn无法直接应用于List.class来忽略其内部特定泛型元素。要忽略List字段,如果可以修改DTO,最佳实践是直接在该字段上使用@JsonIgnore注解。如果DTO不可控,可能需要更复杂的自定义序列化器或反序列化器来处理,或者在克隆前手动将这些字段置空。
  3. 性能考量: 深度克隆涉及对象的序列化和反序列化,对于大型或复杂的对象图,这可能带来一定的性能开销。在性能敏感的场景下,应权衡其利弊或考虑其他克隆策略(如手动属性拷贝)。
  4. Jackson版本: 本文示例基于Jackson Databind 2.13.1。不同版本可能存在细微差异,但核心机制通常保持一致。

总结

当需要深度克隆包含不可控的非序列化类型(如MultipartFile)及其数组的DTO时,Jackson的ObjectMapper结合addMixIn提供了一个强大的解决方案。关键在于理解Jackson对类型匹配的精确性要求,并为原始类型及其数组类型分别配置MixIn。然而,对于泛型集合类型,此方法存在局限性,通常需要结合其他策略来解决。掌握这些技巧,能够帮助开发者在处理复杂DTO克隆场景时更加灵活和高效。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

842

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

742

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

739

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

399

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

431

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

html编辑相关教程合集
html编辑相关教程合集

本专题整合了html编辑相关教程合集,阅读专题下面的文章了解更多详细内容。

16

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.7万人学习

C# 教程
C# 教程

共94课时 | 7.3万人学习

Java 教程
Java 教程

共578课时 | 49.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号