Java中的序列化与反序列化(一)

news/2025/3/15 1:02:15/

1、概述

大家好,我是欧阳方超。今天来看一下Java序列化与反序列化的问题。

2、序列化与反序列化

2.1、序列化与反序列化的概念

在Java中,序列化是将对象转换为可存储或传输的格式(一般为字节流)的过程,序列化后的字节流可以被传输给远程系统,并在那里重新构造成原始对象。反序列化则是将序列化的数据转换回原始对象的过程,反序列化是对象序列化的逆过程,通过反序列化操作能够在接收端恢复出与发送端相同的对象。这是Java中的一种非常重要的机制,可以将对象转换为字节流或其他格式,以便在网络上传输或在磁盘上存储,并在需要的时候进行恢复。

2.2、serialVersionUID出现的缘由

如果在序列化和反序列化前后,类的结构发生了变化(属性有增删),会导致反序列化失败。
为了避免这种情况,Java中引入了serialVersionUID这个机制,用于标识序列化版本。当一个对象被序列化时,会将这个对象的类的serialVersionUID的值写入到字节流中,当一个对象被反序列化时,JVM会从字节流中读取serialVersionUID的值,并将其与反序列化对象所对应类的serialVersionUID值进行比较。如果这两个值不一致,就会抛出InvalidClassException异常,表示反序列化失败。我们用一个例子来验证一下:

import java.io.*;public class TestMain {public static void main(String[] args) {Person person = new Person("Jhon", 20);try {FileOutputStream fileOutputStream = new FileOutputStream("D:\\home\\person.ser");ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);objectOutputStream.writeObject(person);objectOutputStream.close();System.out.println("Serialized data is saved in person.ser");}catch (IOException e) {e.printStackTrace();}}
}class Person implements Serializable {private String name;private int age;public Person(String name, int age) {this.name = name;this.age = age;}public String getName() {return name;}public int getAge() {return age;}
}

上面的例子中,创建了Person类的对象,并经过序列化后保存到了磁盘上,文件名为person.ser,接下来我们为Person类额外添加一个属性address,并试图从磁盘上的person.ser文件恢复出Person类的对象,

import java.io.*;public class TestMain {public static void main(String[] args) {Person person = null;//从文件中反序列化对象try {FileInputStream fileInputStream = new FileInputStream("D:\\home\\person.ser");ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);person = (Person)objectInputStream.readObject();System.out.println(person.getAge());System.out.println(person.getName());} catch (IOException | ClassNotFoundException e) {e.printStackTrace();}}
}class Person implements Serializable {private String name;private int age;private String address;public Person(String name, int age) {this.name = name;this.age = age;}public String getName() {return name;}public int getAge() {return age;}public String getAddress() {return address;}
}

此时是会报错的,

java.io.InvalidClassException: com.Person; local class incompatible: stream classdesc serialVersionUID = -9177105843612582313, local class serialVersionUID = 6433972014701326216at java.io.ObjectStreamClass.initNonProxy(ObjectStreamClass.java:699)at java.io.ObjectInputStream.readNonProxyDesc(ObjectInputStream.java:1885)at java.io.ObjectInputStream.readClassDesc(ObjectInputStream.java:1751)at java.io.ObjectInputStream.readOrdinaryObject(ObjectInputStream.java:2042)at java.io.ObjectInputStream.readObject0(ObjectInputStream.java:1573)at java.io.ObjectInputStream.readObject(ObjectInputStream.java:431)

报错的原因从上面的错误日志也可以看出,是由于“stream classdesc serialVersionUID”与“local class serialVersionUID”导致的错误 。

为了避免这种情况,Java中引入了serialVersionUID这个机制,用于标识序列化版本。当一个对象被序列化时,会将这个对象的类的serialVersionUID的值写入到字节流中,当一个对象被反序列化时,JVM会从字节流中读取serialVersionUID的值,并将其与反序列化对象所对应类的serialVersionUID值进行比较。如果这两个值不一致,就会抛出InvalidClassException异常,表示反序列化失败。
如果一个类没有显式地定义serialVersionUID,则在序列化时会根据类的结构自动生成一个,但是这个自动生成的serialVersionUID值是不可控的。如果类的结构发生了变化,可能会导致自动生成的serialVersionUID与旧版本不一致,从而导致反序列化失败。为了避免这种情况,我们应该显式地定义serialVersionUID,以确保它的唯一性,并在类结构发生变化时手动更新它,以确保类的可序列化和反序列化的兼容性。
下面的示例中,为Person类指定了serialVersionUID,其值为1L,注意,通常我们使用随机数生成器或者手工定义一个值,这样将Person类序列化到本地person.ser文件后,不论怎么为Person类增删属性(待验证的name和age属性就别删了),总能恢复出创建Person对象时为name和age指定的值。

import java.io.*;public class TestMain {public static void main(String[] args) {Person person = new Person("Jhon", 20);try {FileOutputStream fileOutputStream = new FileOutputStream("D:\\home\\person.ser");ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);objectOutputStream.writeObject(person);objectOutputStream.close();System.out.println("Serialized data is saved in person.ser");}catch (IOException e) {e.printStackTrace();}Person person1 = null;//从文件中反序列化对象try {FileInputStream fileInputStream = new FileInputStream("D:\\home\\person.ser");ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);person1 = (Person)objectInputStream.readObject();System.out.println(person1.getAge());System.out.println(person1.getName());} catch (IOException | ClassNotFoundException e) {e.printStackTrace();}// 验证反序列化后的对象与原始对象是否一致System.out.println("Name: " + person1.getName());System.out.println("Age: " + person1.getAge());}
}class Person implements Serializable {private static final long serialVersionUID = 1L;private String name;private int age;public Person(String name, int age) {this.name = name;this.age = age;}public String getName() {return name;}public int getAge() {return age;}
}

3、总结

所以,为了避免反序列失败,为序列化类新增属性时,建议不要修改 serialVersionUID 字段的值,当然如果完全不兼容升级,避免反序列化混乱,那么请修改 serialVersionUID 值。关于序列化和反序列化还有很多其他内容,我们择日继续。
我是欧阳方超,把事情做好了自然就有兴趣了,如果你喜欢我的文章,欢迎点赞、转发、评论加关注。下回见。


http://www.ppmy.cn/news/47730.html

相关文章

为什么网络安全缺口很大,招聘却很少?

2020年我国网络空间安全人才数量缺口超过了140万,就业人数却只有10多万,缺口高达了93%。这里就有人会问了: 1、网络安全行业为什么这么缺人? 2、明明人才那么稀缺,为什么招聘时招安全的人员却没有那么多呢&#xff1…

TensorFlow 2 和 Keras 高级深度学习:6~10

原文:Advanced Deep Learning with TensorFlow 2 and Keras 协议:CC BY-NC-SA 4.0 译者:飞龙 本文来自【ApacheCN 深度学习 译文集】,采用译后编辑(MTPE)流程来尽可能提升效率。 不要担心自己的形象&#x…

C# Lambda表达式

目录 Lambda表达式的语法如下: Lambda表达式的内联特性 Lambda表达式常用的方法 C# Lambda表达式简介 Lambda表达式是C#语言中一种函数式编程的特性,它的主要作用是简化代码和提高代码的可读性。在使用Lambda表达式时,可以通过其内联特性…

【C++STL精讲】stack与queue的基本使用及模拟实现

文章目录 💐专栏导读💐文章导读🌷stack是什么?🌷stack的基本使用🌷stack的模拟实现🌷queue是什么?🌷queue的基本使用🌷queue的模拟实现 💐专栏导读…

SpringBoot起步依赖和自动配置

文章目录 1、起步依赖2、自动配置 1、起步依赖 概念 起步依赖本质上是一个Maven项目对象模型(Project Object Model,POM),定义了对其他库的传递依赖,这些东西加在一起支持某一功能。 简单的说,起步依赖就…

iptables表、链、规则

netfilter/iptables(也就是常说的iptables)组成Linux平台下的包过滤防火墙,具有完成封包过滤、封包重定向和网络地址转换(NAT)等功能。 netfilter是Linux 核心中一个通用架构,它提供了一系列的"表&quo…

手敲Mybatis(九)-结果集处理器

1.前言-背景介绍 上节我们处理了参数处理器,本节我们处理结果集处理器,之前我们写了一个DefaultResultSetHandler,我们把返回结果获取对象,填充值什么的写到了一起,流程没有进行解耦,并且只接收了Object的…

信息系统项目管理师-项目范围管理

1.过程 1.1 规划范围管理 为了记录如何定义、确认和控制项目范围及产品范围,而创建范围管理计划的过程。 1.2 收集需求 为实现目标而确定、记录并管理项目干系人的需要和需求的过程。 1.3 定义范围 制定项目和产品详细描述的过程。 1.4 创建WBS(工作分解…