实现对象之间的序列化和反序列化

1.什么是序列化?

在项目的开发中,为了让前端更好的分析后端返回的结果,我们一般会将返回的信息进行序列化,序列化就是将返回对象的状态信息转换为一种标准化的格式,方便在网络中传输也方便打印日志时号观察,反序列化就是将序列化后的对象还原成原来的对象信息。 

2.传统方式实现序列化 

1.1 对Object类进行序列化和反序列化

我们是通过ObjectMapper中提供的writeValueAsString()方法实现序列化,通过readValue方法实现反序列化。

代码示例:

@SpringBootTest
public class JacksonTest {@Testvoid JacksonTest() throws IOException {ObjectMapper objectMapper=new ObjectMapper();CommonResult<String> commonResult=CommonResult.error(500,"系统错误");//序列化String str=objectMapper.writeValueAsString(commonResult);System.out.println("序列化后:"+str);//反序列化CommonResult commonResult1 = objectMapper.readValue(str.getBytes(StandardCharsets.UTF_8), CommonResult.class);System.out.println("反序列化后:"+commonResult1);
}

运行代码

 

1.2 对List类进行序列化和反序列化

对List的序列化和对Object类的序列化方法一样,不过List类的反序列化需要多一步操作。

  

通过readValue方法的原码发现,第二个参数是Class类型。

由于Java的泛型在编译时会进行类型擦除,这意味着在运行时,泛型的具体类型信息只保留原始类型,例如List<CommonResult>中的CommonResult是通过泛型的形式传递的,编译时CommonResult里的具体信息就会被擦除,运行时就会被视为List,此时List类型中的CommonResult参数的信息可能会丢失或者欠缺了,如果直接将序列化后的List类型直接反序列化为原来的数据,可能会导致CommonResult类型信息丢失,从而引发问题。

所以,此时,我们要构建一个参数化类型,即带有泛型参数的具体类型,以上面为例,我们就需要构建一个List<CommonResult>的参数类型,这个通过下图方法实现

代码实现:

 

@SpringBootTest
public class JacksonTest {@Testvoid JacksonTest() throws IOException {ObjectMapper objectMapper=new ObjectMapper();  String str;//List的序列化List<CommonResult<String>> resultList= Arrays.asList(CommonResult.success("成功1"),CommonResult.success("成功2"));str=objectMapper.writeValueAsString(resultList);System.out.println("序列化后:"+str);//List的反序列化//构造一个带有泛型参数的具体类型JavaType javaType=objectMapper.getTypeFactory().constructParametricType(List.class,CommonResult.class);resultList=objectMapper.readValue(str,javaType);for(CommonResult result:resultList){System.out.println("反序列化后:"+result);}}

在这种实现方式中,我们需要主动处理writeValueAsString和readValue方法抛出的异常,这意味着我们每次调用这两个方法都要处理这两个方法抛出的异常,这是在有点麻烦切有点不美观,如果我们使用的是catch来捕捉异常,这样代码就会变得很难看。

3.模仿springboot原码中实现序列化和反序列化 

 

通过观察原码发现spring中的parseMap和ParseList都是通过tryParse方法来实现反序列化的,tryParse的原码如下图加分析

 

 

所以此时, 我们可以将原码中的tryParse方法复制一份我们模仿实现的代码中,并且用单例模式让ObjectMapper只能被构建一次。

package com.example.lotterysystem.common.utils;import com.fasterxml.jackson.core.JacksonException;
import com.fasterxml.jackson.databind.JavaType;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.springframework.boot.json.JsonParseException;
import org.springframework.util.ReflectionUtils;import java.util.List;
import java.util.concurrent.Callable;public class JacksonUtil {//单例构造要私有Jackson的私有方法private JacksonUtil(){}//单例操作private static final ObjectMapper OBJECT_MAPPER;static {OBJECT_MAPPER=new ObjectMapper();}private static ObjectMapper getObjectMapper(){return OBJECT_MAPPER;}//有队tryParse方法进一步封装,将tryParse捕获的类型写死private static  <T> T tryParse(Callable<T> parser){return tryParse(parser, JacksonException.class);}private static  <T> T tryParse(Callable<T> parser, Class<? extends Exception> check) {try {return parser.call();} catch (Exception var4) {if (check.isAssignableFrom(var4.getClass())) {throw new JsonParseException(var4);}throw new IllegalStateException(var4);}}/*** 序列化方法* @param object* @return*/public static String writeValueAsString(Object object){return JacksonUtil.tryParse(()->{return JacksonUtil.getObjectMapper().writeValueAsString(object);});}/*** 反序列化Object类型数据* @param content* @param valueType* @param <T>* @return*/public static<T> T readValue(String content,Class<T> valueType){return JacksonUtil.tryParse(()->{return JacksonUtil.getObjectMapper().readValue(content,valueType);});}/*** 反序列化List类型* @param content* @param paramClasses* @param <T>* @return*/public static <T> T readListValue(String content,Class<?> paramClasses){JavaType javaType=JacksonUtil.getObjectMapper().getTypeFactory().constructParametricType(List.class,paramClasses);return JacksonUtil.tryParse(()->{return JacksonUtil.getObjectMapper().readValue(content,javaType);});}
}

测试代码

@Testvoid JacksonUtilTest(){CommonResult<String> result=CommonResult.success("成功");String str;//序列化str= JacksonUtil.writeValueAsString(result);System.out.println("序列化后:"+str);//反序列化result=JacksonUtil.readValue(str,CommonResult.class);System.out.println("反序列化后:"+result);System.out.println("下面是List序列化");//List的序列化List<CommonResult<String>> resultList= Arrays.asList(CommonResult.success("成功1"),CommonResult.success("成功2"));str=JacksonUtil.writeValueAsString(resultList);System.out.println("序列化后:"+str);//list的反序列化List<CommonResult<String>> results=JacksonUtil.readListValue(str,CommonResult.class);for(CommonResult son:results){System.out.println("反序列化后:"+son);}}

 

 

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

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

相关文章

ThreadLocal - 原理与应用场景详解

ThreadLocal 的基础概念 在 Java 的多线程世界里&#xff0c;线程之间的数据共享与隔离一直是一个关键话题。如果处理不当&#xff0c;很容易引发线程安全问题&#xff0c;比如数据混乱、脏读等。而 ThreadLocal 这个工具类&#xff0c;就像是为线程量身定制的 “私人储物柜”…

iwebsec靶场 文件包含关卡通关笔记11-ssh日志文件包含

目录 日志包含 1.构造恶意ssh登录命令 2.配置ssh日志开启 &#xff08;1&#xff09;配置sshd &#xff08;2&#xff09;配置rsyslog &#xff08;3&#xff09;重启服务 3.写入webshell木马 4.获取php信息渗透 5.蚁剑连接 日志包含 1.构造恶意ssh登录命令 ssh服务…

Diamond软件的使用--(4)搭建Modelsim仿真库

使用Modelsim仿真的原因 由于diamond自带的仿真软件Active-HDL需要另一套Lisence&#xff0c;所以我们使用第三方仿真软件Modelsim来进行仿真。 Modelsim10.5下载链接如下&#xff1a;https://pan.baidu.com/s/1G9699ocWm1UTqK2yS2igyQ 提取码&#xff1a;lewa 一、Lattice仿…

2025年4月19日,四月第三周,C++,字符串数组答案解析

答案与解析 1. 输出字符串数组所有元素 cpp 复制 下载 #include <iostream> using namespace std;int main() {string arr[] = {"apple", "banana", "cherry"};int n = sizeof(arr)/sizeof(arr[0]); // 计算数组长度for (int i = 0; …

C语言之高校学生信息快速查询系统的实现

&#x1f31f; 嗨&#xff0c;我是LucianaiB&#xff01; &#x1f30d; 总有人间一两风&#xff0c;填我十万八千梦。 &#x1f680; 路漫漫其修远兮&#xff0c;吾将上下而求索。 C语言之高校学生信息快速查询系统的实现 目录 任务陈述与分析 问题陈述问题分析 数据结构设…

【网络篇】TCP vs UDP底层区别+网络编程概念

大家好呀 我是浪前 今天讲解的是网络篇的第三章&#xff1a;网络编程概念和TCP&UDP的区别 网络编程概念TCP和UDP的区别 跨主机通信:网络编程插座&#xff1a;网络编程的本质&#xff1a; 网络编程的重要概念&#xff1a;客户端和服务器&#xff1a; 客户端和服务器的交互模…

EMIF详解

一、EMIF的基本定义 EMIF&#xff08;External Memory Interface&#xff0c;外部存储器接口&#xff09; 是嵌入式处理器&#xff08;如DSP、FPGA、SoC&#xff09;用于连接外部存储器的专用硬件接口模块&#xff0c;负责管理处理器与存储器之间的地址/数据总线、控制信号及时…

Keil MDK 编译问题:function “HAL_IncTick“ declared implicitly

问题与处理策略 问题描述 ..\..\User\stm32f1xx_it.c(141): warning: #223-D: function "HAL_IncTick" declared implicitlyHAL_IncTick(); ..\..\User\stm32f1xx_it.c: 1 warning, 0 errors问题原因 在 stm32f1xx_it.c 文件中调用了 HAL_IncTick()&#xff0c;但…

Java Web项目(一)

框架 java web项目总工分为两部分&#xff1a;客户端&#xff08;前端&#xff09;和服务端&#xff08;后端&#xff09; 客户端发起请求&#xff0c;服务端接受请求并进行处理 发起请求的方式&#xff1a;from表单、jQuery ajax from表单 造成全局的变化&#xff0c;在发…

Dify部署内网时遇到的代理问题及解决办法

大家知道&#xff0c;在公网环境下利用docker安装dify源码镜像比较容易&#xff0c;详见我之前的文章&#xff0c;基于dify开发agent、workflow等非常方便&#xff0c;本次想着在内部网络环境下也完成部署&#xff0c;以方便更多的人使用&#xff0c;但在部署到内网环境下&…

多节点监控的docker管理面板Portainer安装指南:家庭云计算专家

背景 Portainer 是一个轻量级且功能强大的容器管理面板&#xff0c;专为 Docker 和 Kubernetes 环境设计。它通过直观的 Web 界面简化了容器的部署、管理和监控&#xff0c;即使是非技术用户也能轻松上手。Portainer 支持多节点管理&#xff0c;允许用户从一个中央控制台管理多…

Linux内核哈希表学习笔记

前沿 近期项目中需要给自定义的驱动增加一个功能存储相关的数据信息。结合实际业务层面,最终决定采用哈希表的结构来存储。因为其具备快速查找,插入和删除。其实现原理通过散列函数映射到指定位置。时间复杂度O(1).而且运算速度也快,很适合处理大量的数据场景。但是其也有一…

对于在线教育或知识付费类网站视频处理方案

一、视频格式&#xff1a; 1. 推荐格式&#xff1a;HLS&#xff08;HTTP Live Streaming&#xff09; 优势‌&#xff1a; ‌自适应码率‌&#xff1a;根据用户网络状况自动切换清晰度&#xff0c;避免卡顿。‌广泛兼容性‌&#xff1a;iOS/macOS 原生支持&#xff0c;Android…

Deepseek输出的内容如何直接转化为word文件?

我们有时候会直接利用deepseek翻译别人的文章或者想将deepseek输出的内容直接复制到word文档里。但是文本格式和word是不对应的。这时候需要输入如下命令&#xff1a; 以上翻译内容的格式和排版要求如下&#xff1a; 1、一级标题 字体为黑体&#xff08;三号&#xff09;&…

【Vue】组件通信(Props/Emit、EventBus、Provide/Inject)

个人主页&#xff1a;Guiat 归属专栏&#xff1a;Vue 文章目录 1. Props/Emit 父子组件通信1.1 Props 向下传递数据1.2 Emit 向上传递事件 2. EventBus 跨组件通信2.1 创建事件总线2.2 使用事件总线2.3 EventBus 优缺点 3. Provide/Inject 深层组件通信3.1 基本使用3.2 响应式处…

vulnhub sunset系列靶机合集(部分)

描述&#xff1a;该合集包含sunset系列适合新手的四个靶机&#xff08;sunset:1、dusk、sunrise、noontide&#xff09;的渗透全过程。 靶机下载地址&#xff1a;Vulnerable By Design - Search: sunset ~ VulnHubhttps://www.vulnhub.com/?qsunset sunset:1 渗透过程 信息…

【MySQL】MySQL的基础语法及其语句的介绍

1、基础语法 mysql -h【主机名】 -u【用户名】 -p //登录MySQL exit或quit; //退出MySQL show database; //查看MySQL下的所有数据库 use 【数据库名】; //进入数据库 show tables; //查看数据库下的所有表名 *MySQL的启动和关闭 &am…

2025-4-20-C++ 学习 数组(1)

数组 2025-4-20-C++ 学习 数组(1)P1428 小鱼比可爱题目描述输入格式输出格式输入输出样例 #1输入 #1输出 #1说明/提示题解代码P1427 小鱼的数字游戏题目描述输入格式输出格式输入输出样例 #1输入 #1输出 #1说明/提示数据规模与约定题解代码P5727 【深基5.例3】冰雹猜想题目描…

ESP-ADF外设子系统深度解析:esp_peripherals组件架构与核心设计(显示输出类外设之LCD)

目录 ESP-ADF外设子系统深度解析&#xff1a;esp_peripherals组件架构与核心设计&#xff08;显示输出类外设之LCD&#xff09;简介模块概述功能定义架构位置核心特性 LCD外设分析LCD外设概述LCD外设层次架构图 LCD外设API和数据结构外设层API公共API内部数据结构 LCD外设配置选…

面试题:循环引用两个节点相互引用,如何判断哪个用 shared_ptr?哪个用 weak_ptr?

目录 1.引言 2.原理 3.所有权模型与指针选择 4.复杂场景的决策策略 5.注意事项 6.总结 1.引言 当两个对象通过 shared_ptr 相互引用时&#xff0c;会产生循环引用问题&#xff0c;导致内存泄漏。因为这两个对象的引用计数永远不会变为 0&#xff0c;即使它们在程序的其他…