【Java 进阶篇】揭秘 Jackson:Java 对象转 JSON 注解的魔法

在这里插入图片描述

嗨,亲爱的同学们!欢迎来到这篇关于 Jackson JSON 解析器中 Java 对象转 JSON 注解的详细解析指南。JSON(JavaScript Object Notation)是一种常用于数据交换的轻量级数据格式,而 Jackson 作为一款优秀的 JSON 解析库,通过注解提供了便捷而强大的方式,让我们能够更自由地掌控 Java 对象与 JSON 之间的转换。在这篇博客中,我将带你一探 Jackson 注解的奇妙世界,通过示例代码揭秘 Java 对象转 JSON 注解的魔法!

什么是 Jackson 注解?

在开始之前,让我们先简要了解一下 Jackson 注解是什么。

Jackson 注解是一组用于配置 Java 对象序列化和反序列化过程的注解。通过在 Java 类或字段上添加这些注解,我们可以指导 Jackson 如何处理 JSON 转换。这为我们提供了极大的灵活性,让我们能够通过注解方式定制化 JSON 转换过程,满足不同的需求。

基本注解:@JsonProperty

首先,让我们介绍最基本的注解之一:@JsonProperty。这个注解用于指定 JSON 字符串中的字段名与 Java 对象中的字段名之间的映射关系。

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

public class JsonPropertyExample {
    public static void main(String[] args) throws Exception {
        // 创建一个 ObjectMapper 对象
        ObjectMapper objectMapper = new ObjectMapper();

        // 创建一个包含 @JsonProperty 注解的对象
        JsonPropertyObject jsonPropertyObject = new JsonPropertyObject("Alice", 25);

        // 将对象序列化为 JSON 字符串
        String jsonString = objectMapper.writeValueAsString(jsonPropertyObject);

        // 输出结果
        System.out.println(jsonString);
    }
}

在这个例子中,JsonPropertyObject 类的字段 name 被注解为 @JsonProperty("fullName"),这意味着在序列化为 JSON 字符串时,字段 name 将以 "fullName" 作为键。输出结果应该是类似于 {"fullName":"Alice","age":25} 的字符串。

定制化日期格式:@JsonFormat

在处理日期类型时,我们常常需要定制化日期的格式。这时,@JsonFormat 就派上用场了。

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

import java.util.Date;

public class JsonFormatExample {
    public static void main(String[] args) throws Exception {
        // 创建一个 ObjectMapper 对象
        ObjectMapper objectMapper = new ObjectMapper();

        // 创建包含 @JsonFormat 注解的对象
        JsonFormatObject jsonFormatObject = new JsonFormatObject(new Date());

        // 将对象序列化为 JSON 字符串
        String jsonString = objectMapper.writeValueAsString(jsonFormatObject);

        // 输出结果
        System.out.println(jsonString);
    }
}

在这个例子中,JsonFormatObject 类的字段 birthDate 被注解为 @JsonFormat(shape = JsonFormat.Shape.STRING, pattern = "yyyy-MM-dd"),这表示在序列化为 JSON 字符串时,birthDate 将以指定的日期格式呈现。输出结果应该是类似于 {"birthDate":"2023-01-01"} 的字符串。

忽略字段:@JsonIgnore

有时候,我们希望在序列化或反序列化过程中忽略某些字段,这时可以使用 @JsonIgnore 注解。

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

public class JsonIgnoreExample {
    public static void main(String[] args) throws Exception {
        // 创建一个 ObjectMapper 对象
        ObjectMapper objectMapper = new ObjectMapper();

        // 创建包含 @JsonIgnore 注解的对象
        IgnoreFieldObject ignoreFieldObject = new IgnoreFieldObject("Sensitive Data", "Normal Data");

        // 将对象序列化为 JSON 字符串
        String jsonString = objectMapper.writeValueAsString(ignoreFieldObject);

        // 输出结果
        System.out.println(jsonString);
    }
}

在这个例子中,IgnoreFieldObject 类的字段 sensitiveData 被注解为 @JsonIgnore,这表示在序列化为 JSON 字符串时,sensitiveData 字段将被忽略。输出结果应该是类似于 {"normalData":"Normal Data"} 的字符串。

支持枚举:@JsonEnumDefaultValue

在处理枚举类型时,我们可能会遇到枚举值新增但尚未在代码中处理的情况。这时,@JsonEnumDefaultValue 注解可以帮助我们处理未知的枚举值。

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

public class JsonEnumDefaultValueExample {
    public static void main(String[] args) throws Exception {
        // 创建一个 ObjectMapper 对象
        ObjectMapper objectMapper = new ObjectMapper();

        // 创建包含 @JsonEnumDefaultValue 注解的对象
        EnumDefaultValueObject enumDefaultValueObject = new EnumDefaultValueObject(EnumWithDefault.UNKNOWN);

        // 将对象序列化为 JSON 字符串
        String jsonString = objectMapper.writeValueAsString(enumDefaultValueObject);

        // 输出结果
        System.out.println(jsonString);
    }
}

在这个例子中,EnumDefaultValueObject 类的字段 enumValue 被注解为 @JsonEnumDefaultValue,并且指定了默认值为 EnumWithDefault.UNKNOWN。这意味着在序列化为 JSON 字符串时,如果枚举值未知,将使用默认值 UNKNOWN

定制化序列化与反序列化:@JsonSerialize@JsonDeserialize

有时候,我们可能需要对字段进行更复杂的序列化或反序列化操作,这时可以使用 @JsonSerialize@JsonDeserialize 注解。

import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.annotation.JsonDeserialize;
import com.fasterxml.jackson.databind.annotation.JsonSerialize;

public class JsonSerializeDeserializeExample {
    public static void main(String[] args) throws Exception {
        // 创建一个 ObjectMapper 对象
        ObjectMapper objectMapper = new ObjectMapper();

        // 创建包含 @JsonSerialize 和 @JsonDeserialize 注解的对象
        SerializeDeserializeObject serializeDeserializeObject = new SerializeDeserializeObject("customValue");

        // 将对象序列化为 JSON 字符串
        String jsonString = objectMapper.writeValueAsString(serializeDeserializeObject);

        // 输出结果
        System.out.println(jsonString);

        // 将 JSON 字符串反序列化为对象
        SerializeDeserializeObject deserializedObject = objectMapper.readValue(jsonString, SerializeDeserializeObject.class);

        // 输出反序列化结果
        System.out.println(deserializedObject.getCustomValue());
    }
}

在这个例子中,SerializeDeserializeObject 类的字段 customValue 被注解为 @JsonSerialize(using = CustomSerializer.class)@JsonDeserialize(using = CustomDeserializer.class)。这意味着在序列化时将使用自定义的序列化器 CustomSerializer,而在反序列化时将使用自定义的反序列化器 CustomDeserializer

嵌套对象处理:@JsonManagedReference@JsonBackReference

当对象之间存在双向关系时,为了防止无限递归的序列化问题,我们可以使用 @JsonManagedReference@JsonBackReference 注解。

import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.annotation.JsonBackReference;
import com.fasterxml.jackson.databind.annotation.JsonManagedReference;

public class JsonReferenceExample {
    public static void main(String[] args) throws Exception {
        // 创建一个 ObjectMapper 对象
        ObjectMapper objectMapper = new ObjectMapper();

        // 创建包含 @JsonManagedReference 和 @JsonBackReference 注解的对象
        ReferenceParent parent = new ReferenceParent("Parent");
        ReferenceChild child = new ReferenceChild("Child", parent);

        // 设置对象间的关系
        parent.setChild(child);

        // 将对象序列化为 JSON 字符串
        String jsonString = objectMapper.writeValueAsString(parent);

        // 输出结果
        System.out.println(jsonString);
    }
}

在这个例子中,ReferenceParent 类的字段 child 被注解为 @JsonManagedReference,而 ReferenceChild 类的字段 parent 被注解为 @JsonBackReference。这样,序列化时将优先处理 @JsonManagedReference,而忽略 @JsonBackReference,从而避免了无限递归的问题。

小结

通过本文的介绍,我们深入探讨了 Jackson JSON 解析器中 Java 对象转 JSON 注解的强大功能。从基本的 @JsonProperty 到复杂的 @JsonSerialize@JsonDeserialize,再到处理对象间关系的 @JsonManagedReference@JsonBackReference,Jackson 提供了丰富的注解来满足各种需求。希望本文能够帮助你更好地理解和使用 Jackson 注解,让 JSON 转换变得更加得心应手!

作者信息

作者 : 繁依Fanyi
CSDN: https://techfanyi.blog.csdn.net
掘金:https://juejin.cn/user/4154386571867191

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/173382.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

SAP 调取http的x-www-form-urlencoded形式的接口

一、了解下x-www-form-urlencoded形式对于SAP来说有啥区别 简单来说, 1.raw格式就是标准的json格式:{“Name”:“John Smith”,“Age”: 23} 2.x-www格式是要转化一下的:NameJohnSmith&Age23 字段与字段相互连接要用 & 符…

java项目之社区互助平台(ssm+vue)

项目简介 社区互助平台实现了以下功能: 1、一般用户的功能及权限 所谓一般用户就是指还没有注册的过客,他们可以浏览主页面上的信息。但如果有中意的社区互助信息时,要登录注册,只有注册成功才有的权限。2、管理员的功能及权限 用户信息的添…

趣学python编程(七、实现个小网站如此简单 web.py使用介绍)

这里先拿一个小网站的例子来举例,保持好奇心就可以了。因为兴趣才是最好的老师,它能激发人内在的行动力。这里介绍个使用web.py轻量级框架实现的一个小网站,可以看到实现个小网站并不难。python都能用来干什么?那么网站就是它众多…

基于猎食者算法优化概率神经网络PNN的分类预测 - 附代码

基于猎食者算法优化概率神经网络PNN的分类预测 - 附代码 文章目录 基于猎食者算法优化概率神经网络PNN的分类预测 - 附代码1.PNN网络概述2.变压器故障诊街系统相关背景2.1 模型建立 3.基于猎食者优化的PNN网络5.测试结果6.参考文献7.Matlab代码 摘要:针对PNN神经网络…

2023年【A特种设备相关管理(锅炉压力容器压力管道)】模拟考试题及A特种设备相关管理(锅炉压力容器压力管道)作业考试题库

题库来源:安全生产模拟考试一点通公众号小程序 A特种设备相关管理(锅炉压力容器压力管道)模拟考试题参考答案及A特种设备相关管理(锅炉压力容器压力管道)考试试题解析是安全生产模拟考试一点通题库老师及A特种设备相关…

MySQL数据库系统教程

基础篇 通用语法及分类 DDL: 数据定义语言,用来定义数据库对象(数据库、表、字段)DML: 数据操作语言,用来对数据库表中的数据进行增删改DQL: 数据查询语言,用来查询数据库中表的记录DCL: 数据控制语言,用…

【Seata源码学习 】篇三 TM开启全局事务的过程

【Seata源码学习 】篇三 TM开启全局事务的过程 TM发送 单个或批量 消息 以发送GlobalBeginRequest消息为例 TM在执行拦截器链路前将向TC发送GlobalBeginRequest 消息 io.seata.tm.api.DefaultGlobalTransaction#begin(int, java.lang.String) Overridepublic String begin(…

2023.11.20 关于 Spring MVC 详解

目录 MVC 工作流程 Spring MVC 掌握三个功能 创建 Spring MVC 项目 推荐安装插件 EditStarters 安装步骤 使用方法 实现连接功能 基础注解 RequestMapping 指定 GET 和 POST 方法类型 ResponseBody 获取参数 传递 单个 或 多个参数 参数重命名 RequestParam …

P1141 01迷宫(dfs+染色联通块)

染色联通块: 一个格联通的所有格 每个对应的最大可联通格子的个数均相同 分析: 1.只需要计算每个块里的元素个数 2.元素标记对应某个块 3.查找元素时: 由 (1)元素坐标-> (2)查找…

DSP介绍及CCS

文章目录 CCS版本编译器CCS使用注意严禁中文 CCS的基本操作新建工程导入现有工程调整字体的大小工程界面恢复标签的使用 仿真盒小虫子进入在线Debug 芯片TMS320F28355基本介绍特性 DSP中特殊指令dsp指令中的EALLOW EDIS CCS TI官网 版本 CCS版本: CCS8.3.1.0004_…

养猫7年:猫罐头牌子哪个好用?5款口碑好的猫罐头推荐!

猫罐头牌子哪个好用?刚开始养猫真的好心累,因为一开始啥也不懂,关于猫猫的饮食这也不会选那也不会选,就很容易踩雷,为此花了不少钱,相信很多新手铲屎官现在也处于这种状态吧。 作为一个养猫7年的资深铲屎官…

Day01 嵌入式 -----流水灯

一、简单介绍 嵌入式系统中的流水灯是一种常见的示例项目,通常用于演示嵌入式系统的基本功能和控制能力。流水灯由多个发光二极管(LED)组成,这些LED按照一定的顺序依次点亮和熄灭,形成一种像水流一样的流动效果。 二、…

django+drf+vue 简单系统搭建 (3) - 基于类的视图

传统Django中有基于类的视图,Drf中自然也有,目的都是实现功能的模块化继承,封装,减少重复代码。 首先在视图中新增下面代码: # simpletool/views.pyfrom rest_framework.views import APIView from simpletool.seria…

关于使用Java-JWT的笔记

Token的组成规则 一个token分三部分,按顺序为:头部(header),载荷(payload),签证(signature) 由三部分生成token ,三部分之间用“.”号做分隔。 例如:“eyJhbGciOiJIUzI1…

【Android Jetpack】理解ViewModel

文章目录 ViewModel实现ViewModelViewModel的生命周期在Fragments间分享数据ViewModel和SavedInstanceState对比ViewModel原理ViewModel与AndroidViewModel ViewModel Android系统提供控件,比如Activity和Fragment,这些控件都是具有生命周期方法&#x…

对象中扩展运算符的作用

1.对象的合并 let o1 {name: "张三",age: 18,brother: {name: "李四",age: 19,},};//属性不重复let o2 {hobby: "打篮球",};console.log({ ...o1, ...o2 });//属性重复,后面对象的属性会覆盖前面的属性let o3 {name: "王五&q…

P2 C++变量

前言 一 C变量的作用 本期我们来讨论一下c 中的变量。 在一个 C 程序中,大部分内容实际上都是在使用数据。我们操作任何类型的数据,如包括我们想要改变、想要修改, 想要读和写数据。我们都需要把数据存储进叫做变量的东西里面。变量允许我们…

Github搜索技巧

文章目录 1 普通搜索2 高级搜索技巧3 github advance查找工具 1 普通搜索 我们一般在github搜索项目,都是直接在根据仓库关键字搜索项目,可能还会用到图中的匹配条件进行筛选。 这样虽然能实现我们的大部分需求,但还不足实现精确查找。而git…

SpringCloud实用篇02

SpringCloud实用篇02 0.学习目标 1.Nacos配置管理 Nacos除了可以做注册中心,同样可以做配置管理来使用。 1.1.统一配置管理 当微服务部署的实例越来越多,达到数十、数百时,逐个修改微服务配置就会让人抓狂,而且很容易出错。我…

卷积神经网络(ResNet-50)鸟类识别

文章目录 卷积神经网络(CNN)mnist手写数字分类识别的实现卷积神经网络(CNN)多种图片分类的实现卷积神经网络(CNN)衣服图像分类的实现卷积神经网络(CNN)鲜花的识别卷积神经网络&#…