当前位置: 首页 > news >正文

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

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.lryc.cn/news/60983.html

相关文章:

  • 3.函数、结构体、包
  • 科普丨关于 A/B 测试的十问十答
  • 尚融宝——整合OpenFeign与Sentinel实现兜底方法——验证手机号码是否注册功能
  • 几种常见的激活函数
  • MySQL-数据库,数据表的基本操作
  • IC-14W网络IC卡读写器_银河麒麟桌面操作系统V10适配测试报告
  • 面试常见问题
  • matlab数据归一化与反归一化处理
  • 【杂凑算法篇】密码杂凑算法的安全强度
  • 【RobotFramework自动化测试】
  • 操作系统原理 —— 什么是中断?(四)
  • SA168 3BSE003389R1
  • 基于Java+Springboot+Vue+elememt美食论坛平台设计实现
  • Go Etcd
  • 01、Cadence使用记录之新建工程与基础操作(原理图绘制:OrCAD Capture CIS)
  • Redis数据结构与对象-链表和字典
  • 学系统集成项目管理工程师(中项)系列08a_合同管理(上)
  • 【Linux 裸机篇(四)】I.MX6ULL C语言 LED 驱动
  • 我也曾经因安装库而抓狂,直到我遇到了
  • DDPG算法详解
  • 继续学c++
  • Day949.遗留系统之殇:为什么要对遗留系统进行现代化? -遗留系统现代化实战
  • DAY 45 Nginx服务配置
  • 如何收集K8S容器化部署的服务的日志?
  • python删除csv文件中的某几列或行
  • Redis持久化机制导致服务自启动后恢复数据过长无法使用以及如何关闭
  • DAY 37 shell免交互
  • 用python脚本从Cadence导出xdc约束文件
  • 【C++ 六】内存分区、引用
  • markdown基本语法