0

0

Jackson库中实现字段的单向处理:仅反序列化不序列化

心靈之曲

心靈之曲

发布时间:2025-09-20 20:43:01

|

870人浏览过

|

来源于php中文网

原创

Jackson库中实现字段的单向处理:仅反序列化不序列化

本文详细介绍了在Jackson库中如何实现对特定JSON字段的单向处理。通过利用@JsonProperty(access = JsonProperty.Access.WRITE_ONLY)注解,开发者可以确保某个字段在从JSON字符串反序列化为Java对象时被正确读取,而在将Java对象序列化回JSON字符串时则被自动忽略。这在处理REST API响应与请求体中字段行为差异时尤为实用,避免了手动干预,提升了代码的简洁性和健壮性。

Jackson中字段序列化与反序列化挑战

在构建基于restful api的应用程序时,我们经常会遇到这样的场景:从服务器接收的json数据中包含某个标识符(如id),我们需要将其反序列化到java对象中进行业务逻辑处理。然而,在将该java对象作为请求体发送回服务器时,这个id字段可能需要被忽略,因为服务器通常会在创建或更新资源时自动生成或管理id。

例如,当获取一个产品信息时,JSON可能包含id和name:

{
  "id": "1",
  "name": "Product1"
}

但在创建或更新产品时,我们通常只发送name,而id则不应出现在请求体中:

{
  "name": "Product1"
}

Jackson库默认的序列化和反序列化行为是双向的,即一个字段如果能被反序列化,通常也能被序列化。如果直接使用@JsonIgnore,则该字段在序列化和反序列化时都会被忽略,这不符合我们的需求。我们需要一种机制,让id字段在反序列化时有效,而在序列化时无效。

解决方案:@JsonProperty(access = JsonProperty.Access.WRITE_ONLY)

Jackson库提供了@JsonProperty注解的access属性,可以精确控制字段的序列化和反序列化行为。针对上述需求,我们可以使用JsonProperty.Access.WRITE_ONLY。

JsonProperty.Access.WRITE_ONLY的含义是:

  • 反序列化(Deserialization):当JSON字符串转换为Java对象时,Jackson会读取对应的JSON字段并将其值写入Java对象的该属性中。
  • 序列化(Serialization):当Java对象转换为JSON字符串时,Jackson会忽略该属性,不会将其包含在生成的JSON中。

简而言之,对于Jackson处理器而言,这个字段是“只写”的,即只能从JSON写入到Java对象,而不能从Java对象读取并写入到JSON。

实战示例

下面通过一个具体的Product类示例来演示如何使用@JsonProperty(access = JsonProperty.Access.WRITE_ONLY)来实现单向处理。

Chromox
Chromox

Chromox是一款领先的AI在线生成平台,专为喜欢AI生成技术的爱好者制作的多种图像、视频生成方式的内容型工具平台。

下载

首先,定义Product类:

import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;

public class Product {

    // id 字段在反序列化时可读(从JSON写入对象),但在序列化时将被忽略(不从对象写入JSON)
    @JsonProperty(access = JsonProperty.Access.WRITE_ONLY)
    private String id;
    private String name;

    // 构造函数
    public Product() {}

    public Product(String id, String name) {
        this.id = id;
        this.name = name;
    }

    // Getter 和 Setter 方法
    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

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

    public static void main(String[] args) throws Exception {
        ObjectMapper mapper = new ObjectMapper();
        // 启用美化输出,方便查看
        mapper.enable(SerializationFeature.INDENT_OUTPUT);

        // 1. 从包含id的JSON字符串反序列化为Java对象
        String incomingJson = """
            {
                "id": "123",
                "name": "Laptop Pro"
            }
            """;

        System.out.println("--- 反序列化测试 ---");
        Product product = mapper.readValue(incomingJson, Product.class);
        System.out.println("反序列化后的Java对象: " + product);
        // 验证id是否被正确读取
        System.out.println("Java对象中的ID: " + product.getId()); 

        // 2. 将Java对象序列化回JSON字符串
        // 此时,product对象中的id值为"123"
        System.out.println("\n--- 序列化测试 ---");
        String resultJson = mapper.writeValueAsString(product);
        System.out.println("序列化后的JSON字符串:\n" + resultJson);
    }
}

运行结果:

--- 反序列化测试 ---
反序列化后的Java对象: Product{id='123', name='Laptop Pro'}
Java对象中的ID: 123

--- 序列化测试 ---
序列化后的JSON字符串:
{
  "name" : "Laptop Pro"
}

从输出结果可以看到,当我们将包含id的JSON字符串反序列化为Product对象时,id字段被成功读取并赋值给Java对象的id属性。然而,当我们将这个Product对象序列化回JSON字符串时,id字段被自动忽略,生成的JSON中只包含name字段。这完美地实现了我们对字段的单向处理需求。

JsonProperty.Access 枚举详解

JsonProperty.Access枚举提供了四种模式,用于更细粒度地控制字段的访问权限:

  • READ_WRITE (默认):字段在序列化和反序列化时都有效。这是Jackson的默认行为。
  • READ_ONLY:字段只在序列化时有效(从Java对象读取到JSON),在反序列化时会被忽略(不从JSON写入到Java对象)。
    • 例如,一个计算属性,只希望在输出JSON中显示,而不允许外部通过JSON修改。
  • WRITE_ONLY:字段只在反序列化时有效(从JSON写入到Java对象),在序列化时会被忽略(不从Java对象读取到JSON)。
    • 本文示例中使用的就是这种模式,适用于API请求体中包含、响应体中不包含的字段。
  • AUTO:Jackson根据字段的可见性、是否存在Getter/Setter方法等自动推断访问权限。在大多数情况下,AUTO的行为与READ_WRITE相似,但提供了更大的灵活性,尤其是在与@JsonAutoDetect等注解配合使用时。

与@JsonIgnore的区别:@JsonIgnore注解会使Jackson在序列化和反序列化时都完全忽略该字段,除非通过其他方式(如@JsonInclude、@JsonSetter等)明确覆盖。而@JsonProperty(access = ...)则提供了更精细的控制,允许在序列化和反序列化之间选择性地启用或禁用字段。当只需要在单向操作中忽略字段时,access属性是更优的选择。

应用场景与注意事项

  1. REST API设计
    • 创建/更新资源:请求体中可能包含一些只用于写入(如密码、临时ID)的字段,这些字段不应在响应中返回。
    • 响应体优化:响应体中可能包含一些只用于读取(如服务器生成的ID、状态)的字段,这些字段不应在客户端发送的请求体中出现。
  2. 数据模型差异处理:当一个Java对象模型需要适应不同外部系统(如数据库、消息队列、不同版本的API)对JSON字段的不同期望时,access属性非常有用。
  3. 与其他Jackson注解的协同:@JsonProperty(access = ...)可以与其他Jackson注解(如@JsonFormat, @JsonInclude等)协同工作,共同构建复杂的数据映射逻辑。
  4. 可读性与维护性:明确使用access属性可以提高代码的可读性,清晰地表达字段的预期行为,减少因默认行为误解而导致的错误。

总结

@JsonProperty(access = JsonProperty.Access.WRITE_ONLY)是Jackson库中一个强大且实用的特性,它为开发者提供了对JSON字段序列化和反序列化行为的精细控制。通过合理利用此注解,我们可以优雅地处理API设计中常见的字段单向处理需求,避免了手动干预和复杂的自定义序列化器/反序列化器,从而提升了代码的简洁性、健壮性和可维护性。在设计和实现与JSON交互的Java应用程序时,深入理解并善用JsonProperty.Access枚举将大大提高开发效率。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的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数据格式相关文章,帮助大家解决问题。

457

2023.08.07

json是什么
json是什么

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

547

2023.08.23

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

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

335

2023.10.13

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

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

82

2025.09.10

mysql标识符无效错误怎么解决
mysql标识符无效错误怎么解决

mysql标识符无效错误的解决办法:1、检查标识符是否被其他表或数据库使用;2、检查标识符是否包含特殊字符;3、使用引号包裹标识符;4、使用反引号包裹标识符;5、检查MySQL的配置文件等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

210

2023.12.04

Python标识符有哪些
Python标识符有哪些

Python标识符有变量标识符、函数标识符、类标识符、模块标识符、下划线开头的标识符、双下划线开头、双下划线结尾的标识符、整型标识符、浮点型标识符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

322

2024.02.23

java标识符合集
java标识符合集

本专题整合了java标识符相关内容,想了解更多详细内容,请阅读下面的文章。

292

2025.06.11

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

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

76

2026.03.11

热门下载

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

精品课程

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

共23课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.2万人学习

Java 教程
Java 教程

共578课时 | 81.2万人学习

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

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