0

0

Jackson处理多别名JSON字段时优先选择非空值的策略与实践

心靈之曲

心靈之曲

发布时间:2025-12-05 17:39:14

|

523人浏览过

|

来源于php中文网

原创

Jackson处理多别名JSON字段时优先选择非空值的策略与实践

本教程旨在解决jackson反序列化中,当json数据包含多个别名字段且需优先选择其中非空值的问题。文章详细介绍了两种有效的策略:一是通过定义多个智能setter方法,利用`@jsonsetter`注解实现按需更新;二是通过自定义converter结合辅助pojo,将数据转换逻辑与领域模型分离。这两种方法都能实现灵活且健壮的数据映射,有效处理冗余数据,确保数据解析的准确性。

在处理来自第三方系统或存在历史遗留问题的JSON数据时,我们经常会遇到同一个逻辑字段在JSON中以多种名称(别名)出现的情况。更复杂的是,这些别名字段中可能只有一个包含有效的非空值,而其他字段则为null或空字符串。Jackson的@JsonAlias注解虽然能识别多个别名,但它通常无法智能地优先选择非空值,导致反序列化结果不符合预期。本文将深入探讨两种解决方案,帮助开发者在Jackson中实现这种智能的非空值优先选择机制。

方案一:利用智能Setter方法处理多别名与非空值选择

此方案的核心思想是为每个可能的别名定义一个独立的setter方法,并通过@JsonSetter注解将其映射到相应的JSON字段。所有这些setter方法最终都将调用一个核心的setter逻辑,该逻辑负责判断当前字段是否已经存在有效值(非空且非空字符串),并仅在当前字段为空时才接受新的值。

实现原理

  1. 定义核心Setter: 创建一个标准的setter方法,例如setName(String name)。在该方法内部,添加一个条件判断,只有当当前name字段为null或空字符串时,才将其更新为传入的新值。
  2. 定义别名Setter: 为每个别名(如full_name、fullName)创建额外的setter方法,例如setName1(String name)、setName2(String name)。这些方法也使用@JsonSetter注解分别映射到对应的JSON字段。
  3. 委托调用: 别名setter方法不包含业务逻辑,它们直接调用核心setter方法,将接收到的值传递过去。

示例代码

import com.fasterxml.jackson.annotation.JsonSetter;
import com.fasterxml.jackson.databind.ObjectMapper;

import java.util.Arrays;
import java.util.Objects;
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") // 映射到JSON字段"name"
    public void setName(String name) {
        // 只有当当前name字段为null或空时,才更新
        if (NULL_OR_EMPTY.test(this.name)) {
            this.name = name;
        }
    }

    // 别名setter 1,映射到JSON字段"full_name"
    @JsonSetter("full_name")
    public void setName1(String name) {
        setName(name); // 委托给核心setter
    }

    // 别名setter 2,映射到JSON字段"fullName"
    @JsonSetter("fullName")
    public void setName2(String name) {
        setName(name); // 委托给核心setter
    }

    // Getter方法
    public String getName() {
        return name;
    }

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

    public static void main(String[] args) throws Exception {
        // 示例JSON数据,其中"name"为null,"full_name"为空字符串,"fullName"为有效值
        String json = "{ \"name\" : null, \"full_name\" : \"\", \"fullName\" : \"name\"}";

        ObjectMapper mapper = new ObjectMapper();
        MyPojo myPojo = mapper.readValue(json, MyPojo.class);
        System.out.println(myPojo);

        // 示例2:第一个有效值
        json = "{ \"name\" : \"first\", \"full_name\" : \"\", \"fullName\" : \"name\"}";
        myPojo = mapper.readValue(json, MyPojo.class);
        System.out.println(myPojo);

        // 示例3:所有都为空或null
        json = "{ \"name\" : null, \"full_name\" : \"\", \"fullName\" : null}";
        myPojo = mapper.readValue(json, MyPojo.class);
        System.out.println(myPojo);
    }
}

输出:

MyPojo{name='name'}
MyPojo{name='first'}
MyPojo{name='null'}

注意事项与优缺点

  • 优点: 实现直接,易于理解和调试,无需引入额外的类。
  • 缺点:
    • 代码污染: 领域模型(POJO)中会增加多个冗余的setter方法,使得领域模型不再纯粹,可能违反单一职责原则。
    • 逻辑耦合: Setter方法中包含了数据校验和选择逻辑,而不是简单的赋值操作。
    • 扩展性差: 如果需要处理的别名数量很多,或者有多个字段需要类似处理,POJO会变得非常庞大和难以维护。

方案二:利用自定义Converter实现数据转换

为了保持领域模型的简洁和职责分离,我们可以采用自定义Converter的方式。这种方法将JSON解析为一个辅助POJO,该辅助POJO能够捕获所有别名字段的值。然后,通过一个Converter将这个辅助POJO转换为最终的领域POJO,并在转换过程中实现非空值的选择逻辑。

VWO
VWO

一个A/B测试工具

下载

实现原理

  1. 创建辅助POJO: 定义一个临时的POJO(或Java Record),它包含所有可能的别名字段,并使用@JsonProperty将它们映射到JSON中的相应字段。这个POJO的任务仅仅是接收原始JSON数据。
  2. 创建自定义Converter: 继承StdConverter,其中SOURCE是辅助POJO,TARGET是最终的领域POJO。在convert方法中,实现从辅助POJO到目标POJO的转换逻辑,包括遍历所有别名字段并选择第一个非空/非空字符串的值。
  3. 关联Converter: 在最终的领域POJO上,使用@JsonDeserialize(converter = YourConverter.class)注解,告诉Jackson在反序列化时使用这个自定义Converter。

示例代码

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 lombok.ToString;

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

// 辅助POJO,用于接收所有可能的别名值
// 使用record可以简化代码
record AuxiliaryPojo(
    @JsonProperty("name") String name,
    @JsonProperty("full_name") String name1,
    @JsonProperty("fullName") String name2
) {}

// 自定义Converter,负责将AuxiliaryPojo转换为MyPojo
class AuxiliaryPojoToMyPojo extends StdConverter {
    // 定义一个谓词,用于检查字符串是否非null且非空
    public static final Predicate NOT_NULL_OR_EMPTY = s -> s != null && !s.isEmpty();

    @Override
    public MyPojo convert(AuxiliaryPojo v) {
        // 从辅助POJO中找到第一个非null且非空的name值
        String selectedName = findMatching(v.name(), v.name1(), v.name2());
        // 构建最终的MyPojo实例
        return MyPojo.builder()
                .name(selectedName)
                .build();
    }

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

// 最终的领域POJO,使用Lombok简化代码
@Getter
@Builder
@ToString
@JsonDeserialize(converter = AuxiliaryPojoToMyPojo.class) // 指定自定义Converter
public class MyPojo {
    private String name;

    public static void main(String[] args) throws Exception {
        // 示例JSON数据
        String json = "{ \"name\" : null, \"full_name\" : \"\", \"fullName\" : \"name\"}";

        ObjectMapper mapper = new ObjectMapper();
        MyPojo myPojo = mapper.readValue(json, MyPojo.class);
        System.out.println(myPojo);

        // 示例2:第一个有效值
        json = "{ \"name\" : \"first\", \"full_name\" : \"\", \"fullName\" : \"name\"}";
        myPojo = mapper.readValue(json, MyPojo.class);
        System.out.println(myPojo);

        // 示例3:所有都为空或null
        json = "{ \"name\" : null, \"full_name\" : \"\", \"fullName\" : null}";
        myPojo = mapper.readValue(json, MyPojo.class);
        System.out.println(myPojo);
    }
}

输出:

MyPojo(name=name)
MyPojo(name=first)
MyPojo(name=null)

注意事项与优缺点

  • 优点:
    • 职责分离: 领域POJO保持纯净,不包含任何与JSON解析或数据选择相关的逻辑。
    • 代码整洁: 转换逻辑集中在Converter中,易于管理和测试。
    • 可扩展性: 如果需要处理多个字段的别名选择,可以为每个字段创建独立的findMatching逻辑,或者将通用逻辑抽象出来。
  • 缺点:
    • 引入额外类: 需要额外定义一个辅助POJO和一个Converter类,增加了项目的类数量。
    • 概念开销: 对于初学者来说,理解Converter的工作机制可能需要一定时间。
    • 性能考量: 相比直接的setter方法,Converter会引入额外的对象创建(辅助POJO)和方法调用,但在大多数应用场景中,这种性能开销通常可以忽略不计。

总结

在Jackson处理多别名JSON字段并优先选择非空值的问题上,两种方案各有优势:

  • 智能Setter方法 简单直接,适用于别名数量不多、对领域模型侵入性要求不高的场景。它的缺点是可能导致POJO代码冗余和职责不清。
  • 自定义Converter 提供了更清晰的职责分离和更强的可维护性,特别适合于复杂的转换逻辑、别名字段较多或对领域模型纯净度有高要求的项目。虽然引入了额外的类,但从长远来看,它能使代码结构更优。

开发者应根据项目的具体需求、团队编码规范以及对代码整洁度的要求,选择最适合的解决方案。对于大型或复杂的系统,自定义Converter通常是更推荐的做法。

相关专题

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

430

2023.08.02

java在线网站
java在线网站

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

16926

2023.08.03

AO3中文版入口地址大全
AO3中文版入口地址大全

本专题整合了AO3中文版入口地址大全,阅读专题下面的的文章了解更多详细内容。

1

2026.01.21

热门下载

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

精品课程

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

共23课时 | 2.7万人学习

C# 教程
C# 教程

共94课时 | 7.2万人学习

Java 教程
Java 教程

共578课时 | 48.8万人学习

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

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