0

0

Jackson 处理多别名 JSON 字段:优先选择非空值的策略

霞舞

霞舞

发布时间:2025-12-05 19:05:18

|

695人浏览过

|

来源于php中文网

原创

Jackson 处理多别名 JSON 字段:优先选择非空值的策略

本文旨在解决使用 jackson 反序列化 json 数据时,当多个字段可能表示同一信息,且其中部分字段可能为 `null` 或空字符串时,如何优先选择非空值的挑战。我们将探讨两种核心策略:通过定义多个智能 `setter` 方法并结合 `@jsonsetter` 注解,以及利用自定义 `converter` 结合辅助 pojo 和 `@jsondeserialize` 注解,实现灵活且健壮的数据映射,确保数据完整性。

在 Spring Boot 应用中,使用 Jackson 处理来自第三方服务或遗留系统的 JSON 数据时,我们经常会遇到 JSON 结构冗余的问题。例如,一个逻辑上的“名称”信息可能在 JSON 中以 name、full_name 或 fullName 等多个字段表示。更复杂的是,这些字段可能并非总是同时存在或非空,有时只有一个字段包含有效数据,而其他字段为 null 或空字符串。Jackson 的 @JsonAlias 注解虽然可以处理多个别名,但它通常不会智能地忽略 null 或空值,而是按照其内部优先级选择一个值,这可能导致最终 POJO 字段被 null 或空值覆盖。

本教程将详细介绍两种有效策略,以确保 Jackson 在这种场景下能够优先选择非 null 或非空字符串的字段值。

1. 使用多个 @JsonSetter 方法

第一种方法是为每个可能包含目标信息的 JSON 字段定义一个独立的 setter 方法,并使用 @JsonSetter 注解将其映射到对应的 JSON 属性名。核心思想是在这些 setter 方法中引入逻辑,仅当目标 POJO 字段当前为 null 或空时才更新其值。

实现步骤

  1. 定义判空/判空字符串的 Predicate: 为了避免重复逻辑,可以定义一个静态的 Predicate 来检查字符串是否为 null 或空。
  2. 主 setter 方法: 为 POJO 字段定义一个标准 setter,并在其中包含更新逻辑。
  3. 辅助 setter 方法: 为每个别名 JSON 字段定义一个额外的 setter 方法,使用 @JsonSetter 注解将其映射到对应的 JSON 属性,并委托给主 setter 方法。

示例代码

import com.fasterxml.jackson.annotation.JsonSetter;
import com.fasterxml.jackson.databind.ObjectMapper;
import java.util.function.Predicate;

public class MyPojo {
    // 定义一个谓词,用于检查字符串是否为null或空
    public static final Predicate NULL_OR_EMPTY =
        s -> s == null || s.isEmpty();

    private String name;

    // 主setter方法,包含更新逻辑
    @JsonSetter("name")
    public void setName(String name) {
        // 只有当当前name字段为null或空时才更新
        if (NULL_OR_EMPTY.test(this.name)) {
            this.name = name;
        }
    }

    // 辅助setter方法,映射到"full_name",并委托给主setName方法
    @JsonSetter("full_name")
    public void setFullNameAlias(String name) {
        setName(name);
    }

    // 辅助setter方法,映射到"fullName",并委托给主setName方法
    @JsonSetter("fullName")
    public void setCamelCaseFullNameAlias(String name) {
        setName(name);
    }

    // Getter和toString方法,便于测试
    public String getName() {
        return name;
    }

    @Override
    public String toString() {
        return "MyPojo{name='" + name + "'}";
    }

    // 示例用法
    public static void main(String[] args) throws Exception {
        String json1 = "{ \"name\" : null, \"full_name\" : \"\", \"fullName\" : \"John Doe\"}";
        String json2 = "{ \"name\" : \"Jane Doe\", \"full_name\" : \"\", \"fullName\" : null}";
        String json3 = "{ \"name\" : null, \"full_name\" : \"Another Name\", \"fullName\" : null}";

        ObjectMapper mapper = new ObjectMapper();

        MyPojo myPojo1 = mapper.readValue(json1, MyPojo.class);
        System.out.println("JSON 1 Output: " + myPojo1); // Expected: MyPojo{name='John Doe'}

        MyPojo myPojo2 = mapper.readValue(json2, MyPojo.class);
        System.out.println("JSON 2 Output: " + myPojo2); // Expected: MyPojo{name='Jane Doe'}

        MyPojo myPojo3 = mapper.readValue(json3, MyPojo.class);
        System.out.println("JSON 3 Output: " + myPojo3); // Expected: MyPojo{name='Another Name'}
    }
}

注意事项与优缺点

  • 优点: 实现相对直接,不需要额外创建辅助类。
  • 缺点: 这种方法在领域模型中引入了“智能 setter”,这可能违反单一职责原则(setter 通常不应包含复杂的验证或条件逻辑)。此外,如果有很多冗余属性,POJO 类会因为额外的 setter 方法而变得臃肿。

2. 使用自定义 Converter

第二种更优雅的解决方案是利用 Jackson 的 Converter 机制。Converter 允许我们在反序列化过程中将一种 POJO 类型转换为另一种 POJO 类型。这种方法将复杂的选择逻辑从领域模型中分离出来,提高了代码的清晰度和可维护性。

艺映AI
艺映AI

艺映AI - 免费AI视频创作工具

下载

实现步骤

  1. 定义辅助 POJO: 创建一个辅助 POJO,其字段直接对应 JSON 中所有可能的冗余属性。这个 POJO 仅仅用于临时捕获原始 JSON 数据。
  2. 创建自定义 Converter: 实现 com.fasterxml.jackson.databind.util.StdConverter 抽象类,指定从辅助 POJO 到目标 POJO 的转换逻辑。在这个 Converter 中,实现选择第一个非 null 非空值的逻辑。
  3. 在目标 POJO 上应用 Converter: 使用 @JsonDeserialize(converter = YourConverter.class) 注解将自定义 Converter 应用到目标 POJO 类上。

Converter 与 Deserializer 的区别

  • Deserializer: 负责根据 JsonParser 中的原始 JSON 令牌构建 POJO 实例。它处理底层 JSON 结构到 Java 对象的映射。
  • Converter: 在一个 POJO 已经反序列化完成后,将其转换为另一个 POJO。它用于数据转换,而不是直接构建对象。

示例代码

import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.annotation.JsonDeserialize;
import com.fasterxml.jackson.databind.util.StdConverter;
import lombok.Builder;
import lombok.Getter;

import java.util.Arrays;
import java.util.function.Predicate;

// 1. 定义辅助POJO,用于捕获所有可能的JSON字段
// 使用record简化数据类定义,需要Java 16+
record AuxiliaryPojo(
    @JsonProperty("name") String name,
    @JsonProperty("full_name") String name1,
    @JsonProperty("fullName") String name2
) {}

// 2. 创建自定义Converter
public class AuxiliaryPojoToMyPojoConverter extends StdConverter {
    public static final Predicate NOT_NULL_OR_EMPTY =
        s -> s != null && !s.isEmpty();

    @Override
    public MyPojo convert(AuxiliaryPojo v) {
        // 在转换逻辑中,查找第一个非null且非空的字符串
        String finalName = findMatching(v.name(), v.name1(), v.name2());
        // 使用Builder模式创建MyPojo实例
        return MyPojo.builder().name(finalName).build();
    }

    // 辅助方法,用于从多个字符串中查找第一个非null非空值
    private String findMatching(String... args) {
        return Arrays.stream(args)
            .filter(NOT_NULL_OR_EMPTY)
            .findFirst()
            .orElse(null); // 如果所有都为null或空,则返回null
    }
}

// 3. 目标POJO,应用Converter
@Getter
@Builder
@JsonDeserialize(converter = AuxiliaryPojoToMyPojoConverter.class)
public static class MyPojo {
    private String name;

    @Override
    public String toString() {
        return "MyPojo{name='" + name + "'}";
    }

    // 示例用法
    public static void main(String[] args) throws Exception {
        String json1 = "{ \"name\" : null, \"full_name\" : \"\", \"fullName\" : \"John Doe\"}";
        String json2 = "{ \"name\" : \"Jane Doe\", \"full_name\" : \"\", \"fullName\" : null}";
        String json3 = "{ \"name\" : null, \"full_name\" : \"Another Name\", \"fullName\" : null}";

        ObjectMapper mapper = new ObjectMapper();

        MyPojo myPojo1 = mapper.readValue(json1, MyPojo.class);
        System.out.println("JSON 1 Output: " + myPojo1); // Expected: MyPojo{name='John Doe'}

        MyPojo myPojo2 = mapper.readValue(json2, MyPojo.class);
        System.out.println("JSON 2 Output: " + myPojo2); // Expected: MyPojo{name='Jane Doe'}

        MyPojo myPojo3 = mapper.readValue(json3, MyPojo.class);
        System.out.println("JSON 3 Output: " + myPojo3); // Expected: MyPojo{name='Another Name'}
    }
}

注意事项与优缺点

  • 优点: 领域模型 MyPojo 保持干净,不包含任何与反序列化逻辑相关的代码。逻辑封装在 Converter 中,职责分离清晰,可维护性高。
  • 缺点: 需要额外定义一个辅助 POJO 和一个 Converter 类,增加了代码量。对于非常简单的场景,可能会显得有些“过度设计”。

总结

当 Jackson 反序列化遇到多别名 JSON 字段且需要优先选择非 null 或非空值时,我们有两种主要策略:

  1. 多智能 setter 方法: 适用于字段数量不多,且不介意在领域模型中引入少量逻辑的场景。优点是实现简单直接,无需额外类。
  2. 自定义 Converter: 适用于字段冗余复杂、需要保持领域模型纯净、追求更高可维护性和解耦的场景。优点是职责分离清晰,代码结构更专业。

通常情况下,推荐使用自定义 Converter 的方法,因为它提供了更强大的灵活性和更好的代码组织结构,尤其是在处理复杂的反序列化逻辑时。根据项目的具体需求和复杂性,选择最适合的策略。

相关专题

更多
java
java

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

841

2023.06.15

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

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

742

2023.07.05

java自学难吗
java自学难吗

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

737

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有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

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

16926

2023.08.03

Java JVM 原理与性能调优实战
Java JVM 原理与性能调优实战

本专题系统讲解 Java 虚拟机(JVM)的核心工作原理与性能调优方法,包括 JVM 内存结构、对象创建与回收流程、垃圾回收器(Serial、CMS、G1、ZGC)对比分析、常见内存泄漏与性能瓶颈排查,以及 JVM 参数调优与监控工具(jstat、jmap、jvisualvm)的实战使用。通过真实案例,帮助学习者掌握 Java 应用在生产环境中的性能分析与优化能力。

19

2026.01.20

热门下载

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

精品课程

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

共23课时 | 2.7万人学习

C# 教程
C# 教程

共94课时 | 7.1万人学习

Java 教程
Java 教程

共578课时 | 48.5万人学习

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

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