Java实现 Serializable 序列化

2024-04-25 13:38

本文主要是介绍Java实现 Serializable 序列化,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

深度理解Java实现 Serializable 序列化

概念

把对象转换为直接序列的过程叫对象的序列化

把字节序列恢复为对象的过程叫对象的反序列化

用途

  1. 对象持久化
  2. 跨网络数据交换,远程过程调用

对象持久化意味着一个对象的生命周期可以不取决于程序是否运行,实现序列化的对象可以生存在程序的调用之间。通过一个序列化的对象写在磁盘中,然后再调用期间恢复这个对象就可以实现对象持久化的效果

序列化可以弥补不同操作系统之间的差异,在某台电脑上创建一个的对象,将其序列化。通过网络传输就可以将其发送到另一台不同操作系统的电脑上,然后在那里准确的组装出这个对象不必关心字节的顺序或者其他细节。所以在向远程对象发送消息时,必须通过对象序列化来传输参数和返回值。

内部实现

让要实现序列化的类实现java.io.Serializable接口

需要注意的是Serializable没有定义任何方法,只是一个标记接口。

对象序列化是基于字节的,因此需要OutputStream和InputStream继承层次结构。

序列化对象过程:在序列化一个对象的过程中,会创建某些OutPutStream对象,将其封装在一个ObjectOutputStream对象内,之后调用writeObject()方法序列化对象。

反序列化对象:创建某些InputStream对象,并将其封装在一个ObjectInputStream 对象内,之后调用readObject()方法反序列化对象。

反序列化最后得到的是一个指向Object的引用,所以必须向下转型为指定类型的对象

序列化控制

默认的序列化机制已经很强大了,它可以自动将对象中的所有字段自动保存和恢复,但这种默认行为有时候不是我们想要的。比如,对于有些字段,它的值可能与内存位置有关,比如默认的hashCode()方法的返回值,当恢复对象后,内存位置肯定变了,基于原内存位置的值也就没有了意义。还有一些字段,可能与当前时间有关,比如表示对象创建时的时间,保存和恢复这个字段就是不正确的。

还有一些情况,如果类中的字段表示的是类的实现细节,而非逻辑信息,那默认序列化也是不适合的。为什么不适合呢?因为序列化格式表示一种契约,应该描述类的逻辑结构,而非与实现细节相绑定,绑定实现细节将使得难以修改,破坏封装。

Java提供了多种定制序列化的机制,主要的有三种,一种是transient关键字,另外一种是实现Externalizable接口代替实现Serializable,还有一种是实现writeObject和readObject方法。

将字段声明为transient,默认序列化机制将忽略该字段,不会进行保存和恢复。
比如上面的第一个实例中,假设我们在进行序列化和反序列化时不需要保存和恢复no字段的信息,那么我们可以在no字段前面加上一个transient修饰符。

private transient String no;

运行程序我们会得到如下的结果:

------------------序列化前--------------
[HashCode:366712642 学号:001 姓名:Ron 班级:Class 001, HashCode:1829164700 学号:002 姓名:Ron2 班级:Class 002]
------------------反序列化后--------------
[HashCode:2133927002 学号:null 姓名:Ron 班级:Class 001, HashCode:1836019240 学号:null 姓名:Ron2 班级:Class 002]

我们可以到no字段的内容反序列化之后变成了null。将字段声明为transient,不是说就不保存该字段了,而是告诉Java默认序列化机制,不要自动保存该字段了。

利用Serializable定义数据类型

在阅读源码时,遇到一些比较抽象的方法,发现他们定义的数据类型都是用Serializable来定义,比如

public void deleteEntriesByIDS(Serializable[] tds);public void deleteEntry(Serializable id);

这样做的原因是它可以接受多种数据类型,比如String,Integer,Long等,他们都实现了Serizlizable接口。同时JDK1,.5以后有了自动拆装箱的特征,所以均可被以上Serializable定义参数方法接受,这样做的目的就是可以节省代码,提高代码利用率。

有的方法也会利用它做返回值,同理

serialVersionUID的作用

序列化的作用是将对象的状态信息转换为可存储过程或传输形式的过程。Java对象是保存在JVM的堆内存里的,也就是说如果JVM堆不存在了,那么对象也就消失。而序列化提供了一种方案是:即使JVM停机的情况下也能把对象保存下来(把Java对象序列化成可存储或传输的形式,比如保存在文件中。下次需要这个对象时就可以从文件中读取出二进制流,再从二进制流中反序列化出对象)

虚拟机是否允许反序列化,不仅取决于类路径和功能代码是否一致。还有一个非常重要的点是对比两个类的序列化ID(serialVersionUID)是否一致。

如果两个类的serialVersionUID不一致,在反序列化时就会抛出java.io.InvalidClassException,并且指出serialVersionUID不一致。

java.io.InvalidClassException: com.hollis.User1; local class incompatible: stream classdesc serialVersionUID = 1, local class serialVersionUID = 2

这是因为,在进行反序列化时,JVM会把传来的字节流中的serialVersionUID与本地相应实体类的serialVersionUID进行比较,如果相同就认为是一致的,可以进行反序列化,否则就会出现序列化版本不一致的异常,即是InvalidCastException

所以在《阿里巴巴Java开发手册》中才会有:在兼容性升级中,在修改类的时候,不要修改serialVersionUID的原因。除非是完全不兼容的两个版本。所以,serialVersionUID其实是验证版本一致性的。

如果一个类实现了Serializable接口,就必须手动添加一个private static final long serialVersionUID变量,并且设置初始值。如果不添加的话,系统会自己添加一个serialVersionUID,只要类稍加改变,就会重新生成,导致版本不兼容

这篇关于Java实现 Serializable 序列化的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/934832

相关文章

springboot集成easypoi导出word换行处理过程

《springboot集成easypoi导出word换行处理过程》SpringBoot集成Easypoi导出Word时,换行符n失效显示为空格,解决方法包括生成段落或替换模板中n为回车,同时需确... 目录项目场景问题描述解决方案第一种:生成段落的方式第二种:替换模板的情况,换行符替换成回车总结项目场景s

SpringBoot集成redisson实现延时队列教程

《SpringBoot集成redisson实现延时队列教程》文章介绍了使用Redisson实现延迟队列的完整步骤,包括依赖导入、Redis配置、工具类封装、业务枚举定义、执行器实现、Bean创建、消费... 目录1、先给项目导入Redisson依赖2、配置redis3、创建 RedissonConfig 配

SpringBoot中@Value注入静态变量方式

《SpringBoot中@Value注入静态变量方式》SpringBoot中静态变量无法直接用@Value注入,需通过setter方法,@Value(${})从属性文件获取值,@Value(#{})用... 目录项目场景解决方案注解说明1、@Value("${}")使用示例2、@Value("#{}"php

SpringBoot分段处理List集合多线程批量插入数据方式

《SpringBoot分段处理List集合多线程批量插入数据方式》文章介绍如何处理大数据量List批量插入数据库的优化方案:通过拆分List并分配独立线程处理,结合Spring线程池与异步方法提升效率... 目录项目场景解决方案1.实体类2.Mapper3.spring容器注入线程池bejsan对象4.创建

线上Java OOM问题定位与解决方案超详细解析

《线上JavaOOM问题定位与解决方案超详细解析》OOM是JVM抛出的错误,表示内存分配失败,:本文主要介绍线上JavaOOM问题定位与解决方案的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一、OOM问题核心认知1.1 OOM定义与技术定位1.2 OOM常见类型及技术特征二、OOM问题定位工具

Python的Darts库实现时间序列预测

《Python的Darts库实现时间序列预测》Darts一个集统计、机器学习与深度学习模型于一体的Python时间序列预测库,本文主要介绍了Python的Darts库实现时间序列预测,感兴趣的可以了解... 目录目录一、什么是 Darts?二、安装与基本配置安装 Darts导入基础模块三、时间序列数据结构与

基于 Cursor 开发 Spring Boot 项目详细攻略

《基于Cursor开发SpringBoot项目详细攻略》Cursor是集成GPT4、Claude3.5等LLM的VSCode类AI编程工具,支持SpringBoot项目开发全流程,涵盖环境配... 目录cursor是什么?基于 Cursor 开发 Spring Boot 项目完整指南1. 环境准备2. 创建

Python使用FastAPI实现大文件分片上传与断点续传功能

《Python使用FastAPI实现大文件分片上传与断点续传功能》大文件直传常遇到超时、网络抖动失败、失败后只能重传的问题,分片上传+断点续传可以把大文件拆成若干小块逐个上传,并在中断后从已完成分片继... 目录一、接口设计二、服务端实现(FastAPI)2.1 运行环境2.2 目录结构建议2.3 serv

C#实现千万数据秒级导入的代码

《C#实现千万数据秒级导入的代码》在实际开发中excel导入很常见,现代社会中很容易遇到大数据处理业务,所以本文我就给大家分享一下千万数据秒级导入怎么实现,文中有详细的代码示例供大家参考,需要的朋友可... 目录前言一、数据存储二、处理逻辑优化前代码处理逻辑优化后的代码总结前言在实际开发中excel导入很

Spring Security简介、使用与最佳实践

《SpringSecurity简介、使用与最佳实践》SpringSecurity是一个能够为基于Spring的企业应用系统提供声明式的安全访问控制解决方案的安全框架,本文给大家介绍SpringSec... 目录一、如何理解 Spring Security?—— 核心思想二、如何在 Java 项目中使用?——