一篇搞懂!Java对象序列化与反序列化的底层逻辑

简介: 本文介绍了Java中的序列化与反序列化,包括基本概念、应用场景、实现方式及注意事项。序列化是将对象转换为字节流,便于存储和传输;反序列化则是将字节流还原为对象。文中详细讲解了实现序列化的步骤,以及常见的反序列化失败原因和最佳实践。通过实例和代码示例,帮助读者更好地理解和应用这一重要技术。



Hello!我是小米,今天我们来聊聊“序列化与反序列化”!在Java开发中,这两位可是我们的“老朋友”了!不管是存储数据还是远程传输对象的过程中,序列化与反序列化都起到了至关重要的作用。接下来我们就从序列化的定义、使用场景、实现方式及注意事项几个方面,深挖这个话题!

序列化与反序列化的基本概念

首先,什么是序列化?

序列化,就是将对象的状态转换成字节流的过程,这样以后可以通过这些字节流重新构建具有相同状态的对象。换句话说,序列化是将Java对象转换为便于存储和传输的格式。

我们平时操作的对象大多是在内存中运行的,意味着只要程序终止或断电,内存中的数据就会丢失。而序列化为我们提供了一个“把对象变成字节流”的功能,能把内存中的对象保存到硬盘、数据库或者网络中,也就实现了数据的持久化

反序列化,则是序列化的逆过程,即将字节流转换回原Java对象,恢复出原来对象的状态和数据。通过反序列化,存储的数据可以再次用于程序的运行。

我们再通过一个例子更好地理解这个过程:

假设你创建了一个Person类的对象:

通过序列化,我们把person对象转换成一连串的字节,并将这些字节存储在文件中;当我们想再次使用这个person对象时,通过反序列化的方式,将文件中的字节读取并重建出原始的Person对象。是不是很神奇!

序列化与反序列化的应用场景

1. 数据的持久化

序列化可以将数据以字节流的形式永久存储到硬盘中。比如将对象保存到文件中,或用于缓存(如Redis的RDB备份)。

比如,我们可以将应用中的数据持久化存储到数据库或文件中,下次启动程序时将数据恢复到内存中,就不需要重新初始化数据了。这在数据较大或初始化过程复杂时尤为方便。

2. 对象的网络传输

序列化是远程通信的重要手段。将对象序列化后,可以在网络上传输字节流,接收端再反序列化成原始对象。这是RMI、Web服务等实现网络通信的基础。

例如,微服务架构中,多个系统间的数据传递就可以利用序列化传递对象。序列化的对象字节流不依赖于进程环境,能保证Java对象跨网络传输的完整性。

如何实现序列化与反序列化

在Java中,序列化和反序列化是通过实现 Serializable 接口来完成的。

1. 实现 Serializable 接口

对于一个需要序列化的Java类,必须实现 Serializable 接口:

这个接口本身没有任何方法,它只是一个标记接口,用于告诉Java虚拟机这个类可以被序列化。我们也通常会为每个类指定 serialVersionUID,以便在序列化和反序列化过程中保持一致。

2. 使用 ObjectOutputStream 序列化对象

将对象写入文件(序列化)时,可以使用 ObjectOutputStream 类:

在这个例子中,我们将 person 对象序列化并存储在本地文件 person.ser 中。

3. 使用 ObjectInputStream 反序列化对象

将序列化的文件转换回对象(反序列化)时,可以使用 ObjectInputStream 类:

运行这个代码可以将 person.ser 文件反序列化为Person对象,并重新获得其属性值。

序列化与反序列化的优点

  • 数据持久化:实现数据的持久化保存,程序重启后可以恢复之前的对象状态。
  • 远程通信:使得对象可以在不同的进程间或网络上传输,保证数据在跨平台、跨进程之间的兼容性。
  • 缓存:通过序列化将对象缓存到本地文件或内存数据库(如Redis、Memcached)中,读取速度更快。

反序列化失败的原因

在反序列化过程中,有时会遇到反序列化失败的情况,常见的原因之一是:

serialVersionUID 不一致

每个序列化的类都会包含一个 serialVersionUID,它是唯一标识类版本的“身份证”。在反序列化时,Java虚拟机会检查对象的 serialVersionUID 与本地类的 serialVersionUID 是否匹配。如果不一致,Java会抛出 InvalidClassException 异常,表示反序列化失败。

所以,为了保证反序列化的顺利进行,建议在类定义时显式定义 serialVersionUID,例如:

总结:如果你修改了类的结构(比如添加或删除字段),要么重新序列化对象,要么手动设置 serialVersionUID 保持不变。

序列化的最佳实践

  • 显式声明 serialVersionUID:避免在类结构改变时导致反序列化失败。
  • 避免对敏感数据进行序列化:对于一些敏感信息(如密码),可以使用 transient 关键字声明,防止这些字段被序列化。
  • 使用可靠的序列化工具:在一些高并发场景下,传统的Java序列化效率可能不高,可以考虑使用 Google Protocol Buffers、Kryo、Apache Avro 等更高效的序列化工具。

END

序列化和反序列化使得Java对象可以在不同的程序、不同的时间、不同的环境中持久化和传输,解决了对象状态持久保存和网络传输的问题。在实际开发中,灵活运用序列化可以极大提升系统的可扩展性和数据持久性。

希望今天这篇文章能帮你更好地理解序列化和反序列化,赶紧动手试试序列化和反序列化操作吧!

我是小米,一个喜欢分享技术的29岁程序员。如果你喜欢我的文章,欢迎关注我的微信公众号软件求生,获取更多技术干货!

相关文章
|
4月前
|
设计模式 网络协议 数据可视化
Java 设计模式之状态模式:让对象的行为随状态优雅变化
状态模式通过封装对象的状态,使行为随状态变化而改变。以订单为例,将待支付、已支付等状态独立成类,消除冗长条件判断,提升代码可维护性与扩展性,适用于状态多、转换复杂的场景。
453 0
|
4月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
252 1
|
4月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
268 1
|
6月前
|
缓存 安全 Java
Java反射机制:动态操作类与对象
Java反射机制是运行时动态操作类与对象的强大工具,支持获取类信息、动态创建实例、调用方法、访问字段等。它在框架开发、依赖注入、动态代理等方面有广泛应用,但也存在性能开销和安全风险。本文详解反射核心API、实战案例及性能优化策略,助你掌握Java动态编程精髓。
|
6月前
|
存储 人工智能 JavaScript
Java从作用域到对象高级应用​
本内容详细讲解了JavaScript中的作用域类型(函数作用域、块作用域、全局作用域)、作用域链、垃圾回收机制、闭包、变量提升、函数参数、数组方法、内置构造函数、对象高级知识、原型链、对象赋值、深浅拷贝、递归、异常处理及this指向等内容,全面覆盖JS核心概念与编程技巧。
77 0
|
7月前
|
存储 安全 Java
深入理解Java序列化接口及其实现机制
记住,序列化不仅仅是把对象状态保存下来那么简单,它涉及到类的版本控制、安全性和性能等多个重要方面。正确理解和实现Java序列化机制对于构建高效、安全和可维护的Java应用至关重要。
243 0
|
8月前
|
Java 数据库连接 API
Java 对象模型现代化实践 基于 Spring Boot 与 MyBatis Plus 的实现方案深度解析
本文介绍了基于Spring Boot与MyBatis-Plus的Java对象模型现代化实践方案。采用Spring Boot 3.1.2作为基础框架,结合MyBatis-Plus 3.5.3.1进行数据访问层实现,使用Lombok简化PO对象,MapStruct处理对象转换。文章详细讲解了数据库设计、PO对象实现、DAO层构建、业务逻辑封装以及DTO/VO转换等核心环节,提供了一个完整的现代化Java对象模型实现案例。通过分层设计和对象转换,实现了业务逻辑与数据访问的解耦,提高了代码的可维护性和扩展性。
326 1
|
7月前
|
存储 Java
Java对象的内存布局
在HotSpot虚拟机中,Java对象的内存布局分为三部分:对象头(Header)、实例数据(Instance Data)和对齐填充(Padding)。对象头包含Mark Word、Class对象指针及数组长度;实例数据存储对象的实际字段内容;对齐填充用于确保对象大小为8字节的整数倍。
145 0
|
存储 Java
【IO面试题 四】、介绍一下Java的序列化与反序列化
Java的序列化与反序列化允许对象通过实现Serializable接口转换成字节序列并存储或传输,之后可以通过ObjectInputStream和ObjectOutputStream的方法将这些字节序列恢复成对象。
|
8月前
|
JSON Java 数据库连接