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

news2025/1/15 17:43:21

在这里插入图片描述

嗨,亲爱的同学们!欢迎来到这篇关于 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

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于SSM的社区生鲜商城的设计与实现

末尾获取源码 开发语言:Java Java开发工具:JDK1.8 后端框架:SSM 前端:采用JSP技术开发 数据库:MySQL5.7和Navicat管理工具结合 服务器:Tomcat8.5 开发软件:IDEA / Eclipse 是否Maven项目&#x…

OpenAI 超 700 名员工联名逼宫董事会;ChatGPT 新功能“阅后即焚”丨 RTE 开发者日报 Vol.89

开发者朋友们大家好: 这里是 「RTE 开发者日报」 ,每天和大家一起看新闻、聊八卦。我们的社区编辑团队会整理分享 RTE (Real Time Engagement) 领域内「有话题的 新闻 」、「有态度的 观点 」、「有意思的 数据 」、「有思考的 文…

相对定位、绝对定位、固定定位、绝对定位堆叠顺序

相对定位:相对自己本身进行偏移 CSS语法: position: relative;/*相对自己进行定位*/ top: 10px;/*距离上边*/ left: 10px;/*距离左边*/ 演示图: 绝对定位:默认以浏览器进行定位。如果想依照父盒子定位,需要在父盒子…

SpringBean的配置详解

Bean的基础配置 例如&#xff1a;配置UserDaoImpl由Spring容器负责管理 <beanid"userDao"class"com.xfy.dao.Impl.UserDaoImpl"></bean> 此时存储到Spring容器中的Bean的beanName是userDao&#xff0c;值是UserDaoImpl&#xff0c;可以根据bea…

P2 C++变量

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

万字解析设计模式之桥接模式、外观模式

一、桥接模式 1.1概述 桥接模式是一种结构型设计模式&#xff0c;它的作用是将抽象部分和实现部分分离开来&#xff0c;使它们能够独立地变化。这样&#xff0c;抽象部分和实现部分可以分别进行扩展&#xff0c;而不会相互影响。它是用组合关系代替继承关系来实现&#xff0c;…

如何在AIX操作系统上修改Java环境变量

AIX操作系统是IBM的Unix操作系统&#xff0c;通常用于企业级应用和服务器环境。在AIX上配置Java环境变量是执行Java应用程序和开发Java代码的重要步骤。本文将详细介绍如何在AIX上修改Java环境变量&#xff0c;并提供具体示例来帮助你完成这个任务。 步骤1&#xff1a;确定Java…

vue3组件外使用route

1.vue3组件外使用route 在写vue3项目时&#xff0c;有时候我们会把组件内部分逻辑代码分离到外部js中&#xff0c;然后在组件里通过import导入。此时如果我们要在组件外使用route对象&#xff0c;方式与组件内不同&#xff1a; 组件内&#xff1a; import { useRoute } from…

CANopen权威指南【CAN总线协议】

1这个总线定义是老外发明的。 想要使用&#xff0c;就必须按照协议去配置数据帧。 CIA301和cia402协议&#xff0c;实际就是寄存器地址上某一段的定义。 下载地址&#xff1a; CAN in Automation (CiA): Technical documents 注册下载也是非常快的。【没什么难度】 就是资…

基于Python(Pandas+Pyecharts)实现全国热门旅游景点数据可视化【500010037】

导入模块 import jieba import pandas as pd from collections import Counter from pyecharts.charts import Line,Pie,Scatter,Bar,Map,Grid from pyecharts.charts import WordCloud from pyecharts import options as opts from pyecharts.globals import ThemeType from…

【C++】map multimap

文章目录 1.map介绍2.map的使用3.multimap介绍4.multimap的使用 1.map介绍 map的文档 翻译&#xff1a; map是关联容器&#xff0c;它按照特定的次序(按照key来比较)存储由键值key和值value组合而成的元素。 在map中&#xff0c;键值key通常用于排序和惟一地标识元素&#x…

机器学习8:在病马数据集上进行算法比较(ROC曲线与AUC)

ROC曲线与AUC。使用不同的迭代次数&#xff08;基模型数量&#xff09;进行 Adaboost 模型训练&#xff0c;并记录每个模型的真阳性率和假阳性率&#xff0c;并绘制每个模型对应的 ROC 曲线&#xff0c;比较模型性能&#xff0c;输出 AUC 值最高的模型的迭代次数和 ROC 曲线。 …

编译器优化代码研究

《Effective C》条款21&#xff1a; /** * 结论&#xff1a;对自定义类型对象表达式objA*objB objC; * 定义friend MyInt operator*(const MyInt& lhs,const MyInt& rhs) * 编译器优化后&#xff1a;operator*()函数内直接在调用接收处构造(此处的匿名临时对象)&am…

万宾科技智能井盖传感器的特性一览

在不断发展的智慧城市技术领域&#xff0c;科学技术的创新永无止境。智能井盖传感器是科学进步带来的高科技产品&#xff0c;为促进城市生命线并保障地上地下连接点安全提供保障。它就在我们脚下&#xff0c;正在悄然改变城市基础设施和公共服务。智能井盖传感器成为现代城市规…

SD-WAN技术:重新定义网络连接方式

随着数字化转型的不断加速&#xff0c;企业对网络的需求呼之欲出。传统的WAN网络由于配置复杂、成本高昂以及带宽利用率低等问题而面临挑战。这时SD-WAN技术的出现正好派上了用场&#xff0c;通过其虚拟化、自动化和智能化的技术手段&#xff0c;大幅度提高了企业网络性能和可靠…

最新AIGC创作系统ChatGPT系统源码,支持最新GPT-4-Turbo模型,支持DALL-E3文生图,图片对话理解功能

一、AI创作系统 SparkAi创作系统是基于OpenAI很火的ChatGPT进行开发的Ai智能问答系统和Midjourney绘画系统&#xff0c;支持OpenAI-GPT全模型国内AI全模型。本期针对源码系统整体测试下来非常完美&#xff0c;可以说SparkAi是目前国内一款的ChatGPT对接OpenAI软件系统。那么如…

基于晶体结构算法优化概率神经网络PNN的分类预测 - 附代码

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

系统之家U盘重装系统Win10方法步骤

用户发现自己电脑上的Win10系统出现问题了&#xff0c;想要通过重装系统来解决问题。但是&#xff0c;用户还不清楚具体重新安装Win10系统的步骤&#xff0c;接下来小编给大家详细介绍利用U盘完成Win10系统重装的方法&#xff0c;在这里用户需要下载系统之家装机大师软件&#…

Kubeadm部署Kubernetes Containerd集群

文章目录 概述一、硬件系统二、基础配置设置主机名配置主机名与IP地址解析关闭防火墙与selinux时间同步(ntp)升级系统内核配置内核转发及网桥过滤*安装ipset及ipvsadm关闭SWAP分区 三、Containerd准备Containerd获取下载解压Containerd配置文件生成并修改Containerd启动及开机自…

金蝶云星空套打设计平台导出套打模板和导入套打模板

文章目录 金蝶云星空套打设计平台导出套打模板和导入套打模板A环境导出套打模板B环境导入套打模板 金蝶云星空套打设计平台导出套打模板和导入套打模板 A环境导出套打模板 导出后&#xff1a; B环境导入套打模板 不要在已设计好的模板导入&#xff0c;会被覆盖 一定记得&am…