0

0

使用Jackson @JsonMerge实现数据深度合并与部分更新

心靈之曲

心靈之曲

发布时间:2025-11-18 15:52:07

|

862人浏览过

|

来源于php中文网

原创

使用jackson @jsonmerge实现数据深度合并与部分更新

本教程详细探讨了在使用Jackson ObjectReader进行数据更新时,如何避免因JSON请求中缺少字段而导致现有数据被意外覆盖的问题。文章介绍了Jackson 2.9及以上版本引入的@JsonMerge注解,并通过具体代码示例,演示了如何利用该注解实现复杂对象的深度合并,确保在部分更新场景下,未提供的字段能够保留其原始值,从而实现更健壮的数据处理逻辑。

在现代应用开发中,通过RESTful API接收部分更新请求是常见需求。Jackson作为Java生态中广泛使用的JSON处理库,其ObjectReader提供了readerForUpdating()方法,旨在方便地将传入的JSON数据合并到现有对象中。然而,在默认情况下,如果传入的JSON请求中缺少某个字段,readerForUpdating()会将现有对象中对应的字段值设置为null,而非保留其原始值。这种行为在需要进行深度合并(deep merge)的场景下,可能会导致数据意外丢失。

Jackson ObjectReader默认更新行为解析

首先,我们通过一个具体的数据模型和更新场景来理解Jackson的默认行为。

假设我们有以下数据类:

data class Model(
  val fieldTypeA: FieldTypeA? = null,
)

data class FieldTypeA(
  val valueA: String? = null,
  val valueB: String? = null,
)

现在,我们从数据库中读取一个现有对象,其fieldTypeA.valueA字段已赋值:

val existingModel = Model(fieldTypeA = FieldTypeA(valueA = "Test", valueB = "Initial B"))
println("现有对象: $existingModel")
// 输出: 现有对象: Model(fieldTypeA=FieldTypeA(valueA=Test, valueB=Initial B))

接下来,我们准备一个JSON更新请求,该请求仅包含fieldTypeA.valueB字段,而fieldTypeA.valueA字段在JSON中缺失:

{
  "fieldTypeA": {
    "valueB": "I am value B"
  }
}

使用Jackson的ObjectReader.readerForUpdating()方法尝试将此JSON合并到existingModel中:

import com.fasterxml.jackson.databind.ObjectMapper
import com.fasterxml.jackson.module.kotlin.KotlinModule

val mapper = ObjectMapper().registerModule(KotlinModule.Builder().build())
val readerForUpdating = mapper.readerForUpdating(existingModel)

val jsonRequest = """{"fieldTypeA":{"valueB":"I am value B"}}"""
val updatedModel: Model = readerForUpdating.readValue(jsonRequest)

println("更新后的对象: $updatedModel")
// 预期输出 (错误行为): 更新后的对象: Model(fieldTypeA=FieldTypeA(valueA=null, valueB=I am value B))

从上述输出可以看出,fieldTypeA.valueA的值被意外地从"Test"覆盖为null。这显然不是我们期望的深度合并行为,我们希望valueA能够保持其原始值。

网龙b2b仿阿里巴巴电子商务平台
网龙b2b仿阿里巴巴电子商务平台

本系统经过多次升级改造,系统内核经过多次优化组合,已经具备相对比较方便快捷的个性化定制的特性,用户部署完毕以后,按照自己的运营要求,可实现快速定制会费管理,支持在线缴费和退费功能财富中心,管理会员的诚信度数据单客户多用户登录管理全部信息支持审批和排名不同的会员级别有不同的信息发布权限企业站单独生成,企业自主决定更新企业站信息留言、询价、报价统一管理,分系统查看分类信息参数化管理,支持多样分类信息,

下载

利用@JsonMerge实现深度合并

为了解决上述问题,Jackson在2.9及以上版本引入了@JsonMerge注解。该注解允许我们指定在进行对象更新时,如果JSON中对应的字段缺失,Jackson不应简单地将现有值设为null,而是尝试进行深度合并。

要启用深度合并行为,只需在需要合并的复杂对象字段上添加@JsonMerge注解。在我们的示例中,我们需要将@JsonMerge应用到Model类中的fieldTypeA字段上:

import com.fasterxml.jackson.annotation.JsonMerge

data class Model(
  @JsonMerge // 在这里添加 @JsonMerge 注解
  val fieldTypeA: FieldTypeA? = null,
)

data class FieldTypeA(
  val valueA: String? = null,
  val valueB: String? = null,
)

通过添加@JsonMerge注解,Jackson的ObjectReader在处理fieldTypeA字段时,会采取不同的策略。如果传入的JSON中fieldTypeA对象存在,并且其中某个属性(如valueA)缺失,Jackson将保留existingModel中fieldTypeA.valueA的原始值,而不是将其覆盖为null。

实际应用与代码演示

现在,让我们使用带有@JsonMerge注解的Model类,再次执行更新操作:

import com.fasterxml.jackson.databind.ObjectMapper
import com.fasterxml.jackson.module.kotlin.KotlinModule
import com.fasterxml.jackson.annotation.JsonMerge // 确保导入

// 定义带有 @JsonMerge 的数据类
data class ModelWithMerge(
  @JsonMerge
  val fieldTypeA: FieldTypeA? = null,
)

data class FieldTypeA(
  val valueA: String? = null,
  val valueB: String? = null,
)

fun main() {
    val mapper = ObjectMapper().registerModule(KotlinModule.Builder().build())

    val existingModel = ModelWithMerge(fieldTypeA = FieldTypeA(valueA = "Test", valueB = "Initial B"))
    println("现有对象: $existingModel")
    // 输出: 现有对象: ModelWithMerge(fieldTypeA=FieldTypeA(valueA=Test, valueB=Initial B))

    val readerForUpdating = mapper.readerForUpdating(existingModel)

    val jsonRequest = """{"fieldTypeA":{"valueB":"I am value B"}}"""
    val updatedModel: ModelWithMerge = readerForUpdating.readValue(jsonRequest)

    println("更新后的对象: $updatedModel")
    // 预期输出 (正确行为): 更新后的对象: ModelWithMerge(fieldTypeA=FieldTypeA(valueA=Test, valueB=I am value B))
}

运行上述代码,您会发现updatedModel中的fieldTypeA.valueA字段成功地保留了其原始值"Test",而fieldTypeA.valueB则被更新为"I am value B"。这正是我们所期望的深度合并行为。

注意事项与最佳实践

在使用@JsonMerge进行深度合并时,请注意以下几点:

  1. Jackson版本要求:@JsonMerge注解是在Jackson 2.9版本中引入的。因此,确保您的项目依赖的Jackson版本至少为2.9或更高。
  2. 注解位置:@JsonMerge应应用于复杂对象类型的字段上,而不是基本类型(如String, Int等)或集合类型。它指示Jackson在合并这些复杂对象时,应该递归地进行属性合并。
  3. 合并逻辑:@JsonMerge的合并逻辑是针对对象的属性进行的。如果JSON中提供了某个属性的新值,则该属性会被更新;如果某个属性在JSON中缺失,但其父对象(被@JsonMerge注解的字段)存在于现有对象中,则该属性的现有值会被保留。
  4. readerForUpdating()的作用:readerForUpdating()方法的核心在于它接收一个现有对象作为参数,Jackson会尝试将JSON数据应用到这个现有对象上,而不是创建一个全新的对象。@JsonMerge在此基础上提供了更精细的合并控制。
  5. 性能考量:深度合并操作通常比简单的对象替换需要更多的处理开销。在性能敏感的场景下,应评估其影响。

总结

Jackson的@JsonMerge注解为处理复杂对象的局部更新提供了一个强大而灵活的解决方案。通过在数据模型中恰当地使用此注解,开发者可以避免在进行部分更新时,因JSON请求中字段缺失而导致现有数据被意外覆盖的问题。这使得Jackson在构建健壮且符合预期的API更新逻辑方面更加得心应手,特别适用于需要保留部分数据状态的场景。理解并运用@JsonMerge是掌握Jackson高级数据绑定技巧的关键一步。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
PHP API接口开发与RESTful实践
PHP API接口开发与RESTful实践

本专题聚焦 PHP在API接口开发中的应用,系统讲解 RESTful 架构设计原则、路由处理、请求参数解析、JSON数据返回、身份验证(Token/JWT)、跨域处理以及接口调试与异常处理。通过实战案例(如用户管理系统、商品信息接口服务),帮助开发者掌握 PHP构建高效、可维护的RESTful API服务能力。

179

2025.11.26

json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

455

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

546

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

334

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

82

2025.09.10

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1010

2023.08.02

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

254

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

1089

2024.03.01

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

3

2026.03.11

热门下载

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

精品课程

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

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.1万人学习

Java 教程
Java 教程

共578课时 | 80.6万人学习

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

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