0

0

Jackson库中实现JSON字段的单向读写控制

心靈之曲

心靈之曲

发布时间:2025-09-20 23:35:00

|

977人浏览过

|

来源于php中文网

原创

jackson库中实现json字段的单向读写控制

本文详细介绍了如何使用Jackson库实现JSON字段的单向控制,即在反序列化时读取某个字段(如ID),但在序列化时将其忽略。通过 @JsonProperty(access = JsonProperty.Access.WRITE_ONLY) 注解,开发者可以精确控制字段的读写行为,确保数据传输的灵活性和安全性,避免不必要的字段出现在输出JSON中。

1. 引言:JSON字段读写分离的需求

在构建RESTful API时,我们经常会遇到这样的场景:某些数据字段在从客户端接收(反序列化)JSON时是必需的,我们需要读取并处理它们;但在服务器向客户端发送响应(序列化)JSON时,这些字段却不应包含在输出中。

例如,一个产品实体可能包含一个 id 字段。当客户端提交一个新产品或更新产品信息时,这个 id 字段可能用于内部验证或关联现有数据。然而,在服务器返回产品详情时,该 id 可能由服务器内部管理,不希望或不需要回传给客户端,以简化响应结构或遵循特定的API契约。

传统的Jackson注解如 @JsonIgnore 可以完全忽略一个字段,使其在序列化和反序列化时都不被处理。但这无法满足我们对字段进行“只读”(反序列化时读取,序列化时忽略)或“只写”(反序列化时忽略,序列化时包含)的细粒度控制需求。为了解决这一问题,Jackson提供了更强大的机制。

2. 解决方案:@JsonProperty的access属性

Jackson库通过 @JsonProperty 注解的 access 属性,为我们提供了对字段读写行为的精确控制。该属性接受 JsonProperty.Access 枚举类型的值,允许我们定义字段在序列化和反序列化过程中的可见性。

核心枚举值及其作用如下:

  • JsonProperty.Access.WRITE_ONLY:

    • 反序列化 (JSON -> Java对象):该字段会被读取并写入Java对象。
    • 序列化 (Java对象 -> JSON):该字段将被忽略,不会出现在输出的JSON字符串中。
    • 应用场景:最适合本文开头描述的需求,例如,需要接收但无需回传的ID、密码等。
  • JsonProperty.Access.READ_ONLY:

    • 反序列化 (JSON -> Java对象):该字段将被忽略,不会从JSON中读取。
    • 序列化 (Java对象 -> JSON):该字段会被读取并包含在输出的JSON字符串中。
    • 应用场景:适用于由服务器生成且客户端只读的字段,例如创建时间、最后更新时间、只读的计算属性等。
  • JsonProperty.Access.READ_WRITE:

    • 反序列化 & 序列化:该字段在两个方向上都可见并被处理。
    • 应用场景:这是默认行为,适用于大多数常规字段。
  • JsonProperty.Access.AUTO:

    Mapify
    Mapify

    Mapify是由Xmind推出的AI思维导图生成工具,原名ChatMind

    下载
    • Jackson会根据字段的getter/setter方法自动推断其访问权限。通常与 READ_WRITE 行为一致,除非有其他注解明确指定。

3. 实践示例

下面通过一个具体的Java实体类和使用Jackson ObjectMapper 的示例,演示如何利用 JsonProperty.Access.WRITE_ONLY 实现字段的单向控制。

3.1 Java实体类定义

我们定义一个 Product 类,其中 id 字段被标记为 WRITE_ONLY,而 name 字段保持默认的读写行为。

import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.annotation.JsonProperty.Access;

public class Product {
    // id 字段只在反序列化时写入,序列化时忽略
    @JsonProperty(access = Access.WRITE_ONLY)
    public String id; 

    // name 字段默认是 READ_WRITE,双向处理
    public 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 + "'}";
    }
}

3.2 使用示例代码

接下来,我们使用 ObjectMapper 进行反序列化和序列化操作,观察 id 字段的行为。

import com.fasterxml.jackson.databind.ObjectMapper;

public class JacksonAccessDemo {
    public static void main(String[] args) throws Exception {
        ObjectMapper mapper = new ObjectMapper();

        // 模拟接收到的JSON字符串,包含id和name
        String incomingJson = """
            {
                "id": "prod123",
                "name": "Widget A"
            }
            """;

        System.out.println("--- 反序列化 (JSON -> Object) ---");
        // 1. 反序列化:将JSON字符串转换为Product对象
        // 由于id被标记为WRITE_ONLY,它将在反序列化时被正确读取并赋值给Product对象的id字段
        Product product = mapper.readValue(incomingJson, Product.class);
        System.out.println("反序列化后的Product对象: " + product.toString());
        // 验证id是否被读取
        System.out.println("Product对象中的ID: " + product.getId()); 

        System.out.println("\n--- 序列化 (Object -> JSON) ---");
        // 2. 序列化:将Product对象转换为JSON字符串
        // 假设我们修改了name,但id保持不变(或由系统内部生成)
        product.setName("Updated Widget A");
        // 由于id被标记为WRITE_ONLY,它将在序列化时被忽略,不会出现在输出的JSON中
        String outgoingJson = mapper.writeValueAsString(product);
        System.out.println("序列化后的JSON字符串: " + outgoingJson);
    }
}

3.3 运行结果分析

运行上述 JacksonAccessDemo 类,你将看到以下输出:

--- 反序列化 (JSON -> Object) ---
反序列化后的Product对象: Product{id='prod123', name='Widget A'}
Product对象中的ID: prod123

--- 序列化 (Object -> JSON) ---
序列化后的JSON字符串: {"name":"Updated Widget A"}

从输出中我们可以清晰地看到:

  • 在反序列化阶段,id 字段 (prod123) 成功地从 incomingJson 中读取并赋值给了 product 对象的 id 属性。
  • 在序列化阶段,尽管 product 对象的 id 属性仍然持有值 (prod123),但由于 @JsonProperty(access = Access.WRITE_ONLY) 的作用,它在生成 outgoingJson 时被完全忽略,输出的JSON中只包含 name 字段。

这完美地实现了对 id 字段的单向控制:只在反序列化时写入对象,在序列化时从JSON中排除。

4. 注意事项与最佳实践

  • 明确API契约: 在设计API时,应明确哪些字段是只读、只写或读写,并在API文档中详细注明,以避免客户端误解。
  • 安全性考量: 对于敏感信息(如密码哈希、内部审计ID等),应始终使用 WRITE_ONLY 或 @JsonIgnore 来防止意外泄露。
  • 代码可读性: 明确使用 Access 枚举值,增强代码的意图表达,让其他开发者一目了然地理解字段的JSON行为。
  • 与@JsonIgnore的区别 再次强调,@JsonIgnore 会在序列化和反序列化两个方向上完全忽略字段。而 access 属性提供了更细粒度的控制,允许我们选择性地在某个方向上处理字段。
  • 构造函数与Setter: 确保当字段被标记为 WRITE_ONLY 时,反序列化过程能够通过公共的构造函数或Setter方法正确地将JSON值赋给Java对象的该字段。如果字段是私有的且没有Setter,Jackson可能无法反序列化。
  • Jackson版本兼容性: 确保你使用的Jackson库版本支持 JsonProperty.Access 枚举。此功能在Jackson 2.x 版本中广泛可用。

5. 总结

@JsonProperty(access) 是Jackson库中一个非常强大且实用的特性,它允许开发者对JSON字段的序列化和反序列化行为进行精细控制。通过灵活运用 WRITE_ONLY 和 READ_ONLY 等访问模式,我们可以构建出更加健壮、安全且符合业务需求的API,有效管理数据流,避免不必要的信息泄露,并简化API响应结构。掌握这一特性对于任何使用Jackson进行JSON处理的Java开发者都至关重要。

相关专题

更多
java
java

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

826

2023.06.15

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

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

726

2023.07.05

java自学难吗
java自学难吗

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

731

2023.07.31

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

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

396

2023.08.01

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

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

398

2023.08.02

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

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

445

2023.08.02

java有什么用
java有什么用

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

429

2023.08.02

java在线网站
java在线网站

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

16884

2023.08.03

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

150

2025.12.31

热门下载

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

精品课程

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

共23课时 | 2.2万人学习

C# 教程
C# 教程

共94课时 | 5.8万人学习

Java 教程
Java 教程

共578课时 | 40.9万人学习

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

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