JAVA 初识序列化与反序列化

JAVA 初识序列化与反序列化

目录

  • JAVA 初识序列化与反序列化
  • 初识序列化与反序列化
  • 1 概述
  • 2 特点/应用场景
  • 3 涉及到的流对象
  • 4 代码实现序列化与反序列化
    • 4.1 步骤1:创建学生类Student
    • 4.2 步骤2:创建序列化测试类
  • 5 测试报错NotSerializableException:
  • 6 测试报错InvalidClassException:
  • 7 为什么反序列化版本号需要与序列化版本号一致?

初识序列化与反序列化

1 概述

序列化是指将对象的状态信息转换为可以存储或传输形式的过程.在序列化期间,对象将其当前状态写入到临时或持久性存储区.以后可以通过从存储区中读取或者反序列化对象的状态,重新创建该对象.

序列化:利用ObjectOutputStream,把对象的信息,按照固定的格式转成一串字节值输出并持久保存到磁盘
反序列化:利用ObjectInputStream,读取磁盘中之前序列化好的数据,重新恢复成对象
在这里插入图片描述

2 特点/应用场景

需要序列化的文件必须实现Serializable接口,用来启用序列化功能
不需要序列化的数据可以修饰成static,原因:static资源属于类资源,不随着对象被序列化输出
每一个被序列化的文件都有一个唯一的id,如果没有添加此id,编译器会自动根据类的定义信息计算产生一个
在反序列化时,如果和序列化的版本号不一致,无法完成反序列化
常用与服务器之间的数据传输,序列化成文件,反序列化读取数据
常用使用套接字流在主机之间传递对象
不需要序列化的数据也可以被修饰成transient(临时的),只在程序运行期间在内存中存在,不会被序列化持久保存

3 涉及到的流对象

序列化:ObjectOutputStream

ObjectOutputStream 将 Java 对象的基本数据类型写入 OutputStream,通过在流中使用文件可以实现对象的持久存储。如果流是网络套接字流,则可以在另一台主机上或另一个进程中重构对象。

构造方法:
ObjectOutputStream(OutputStream out)
创建写入指定 OutputStream 的 ObjectOutputStream
普通方法:
writeObject(Object obj)
将指定的对象写入 ObjectOutputStream

反序列化:ObjectInputStream
ObjectInputStream对以前使用ObjectOutputStream写入的基本数据和对象进行反序列化重构对象。

构造方法:
ObjectInputStream(InputStream in) 创建从指定 InputStream 读取的 ObjectInputStream
普通方法:
readObject() 从 ObjectInputStream 读取对象

4 代码实现序列化与反序列化

4.1 步骤1:创建学生类Student

创建包: cn.tedu.serializable
创建类: Student.java
package cn.tedu.serializable;

import java.io.Serializable;/**本类用来封装学生类*/
/*** 如果本类想要完成序列化,必须实现可序列化接口,否则会报错:* 报错信息:java.io.NotSerializableException: cn.tedu.serializable.Student* Serializable接口是一个空接口,里面一个方法都没有,作用是用来当做标志,标志这个类可以序列化/反序列化* */
public class Student implements Serializable{/**需要给每个进行序列化的文件分配唯一的UID值*///The serializable class Student does not declare a static final serialVersionUID field of type long//private static final long serialVersionUID = 1L;private static final long serialVersionUID = -3193364654654535741L;//1.定义学生的相关属性 + private封装private String name;//姓名private int age;//年龄private String addr;//地址private char gender;//性别/**自动创建构造方法:右键-->Source-->Generate Constructor using Fields...*///2.创建无参构造--必须手动提供无参构造,否则会被含参构造覆盖public Student() {System.out.println("我是Student的无参构造");}//3.创建全参构造public Student(String name,int age,String addr,char gender) {super();//默认调用父类的无参构造this.name = name;this.age = age;this.addr = addr;this.gender = gender;System.out.println("我是Student的全参构造");}//4.属性封装后,需要本类提供公共的属性访问与设置方式get()&set()/**自动创建get()&set(),右键-->Source-->Generate Getters and Setters...*/public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public String getAddr() {return addr;}public void setAddr(String addr) {this.addr = addr;}public char getGender() {return gender;}public void setGender(char gender) {this.gender = gender;}//打印结果:cn.tedu.serializable.Student@4c873330-->地址值//想看对象的属性值,原因是想查看序列化后对象的属性,需要重写toString()//5.重写toString()//自动生成toString()--右键-->Source-->Generate toString()@Overridepublic String toString() {return "Student [name=" + name + ", age=" + age + ", addr=" + addr + ", gender=" + gender + "]";}
}

4.2 步骤2:创建序列化测试类

创建包: cn.tedu.serializable
创建类: TestSerializable.java

package cn.tedu.serializable;import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;/**本类用于序列化与反序列化的测试类*/
//序列化:是指把程序中的java对象,永久保存在磁盘中,相当于是写出的过程,方向是out-->ObjectOutputStream
//反序列化:是指把已经序列化在文件中保存的数据,读取/恢复到java程序中的过程,方向是in-->ObjectInputStream
public class TestSerializable {public static void main(String[] args) {//method();//本方法用来完成序列化的功能method2();//本方法用于完成反序列化功能}/**反序列化方法*/public static void method2() {//声明在本方法内都生效的局部变量,局部变量需要初始化,默认值是nullObjectInputStream in = null;try {//1.创建ObjectInputStream流对象来完成反序列化in = new ObjectInputStream(new FileInputStream("D://ready//1.txt"));//2.通过流对象反序列化生成指定对象Object o = in.readObject();System.out.println(o);System.out.println("恭喜您!反序列化成功!");} catch (Exception e) {System.out.println("很抱歉!反序列化失败!");e.printStackTrace();} finally {//一定会执行的代码块写释放资源的代码try {//3.释放资源in.close();} catch (IOException e) {e.printStackTrace();}}}/**序列化方法*/public static void method() {//声明在本方法内都生效的局部变量,局部变量需要初始化,默认值是nullObjectOutputStream out = null;try {//1.创建ObjectOutputStream流对象来完成序列化out = new ObjectOutputStream(new FileOutputStream("D://ready//1.txt"));//2.指定要序列化(输出)的对象Student obj = new Student("海绵宝宝",3,"大海底部",'男');//3.通过OOS流对象来序列化输出Student对象out.writeObject(obj);System.out.println("恭喜你!序列化成功!");} catch (IOException e) {System.out.println("很抱歉!序列化失败!");e.printStackTrace();}finally {//关流的操作要放在finally{}--因为此代码块一定会执行try {//4.关流操作out.close();} catch (IOException e) {e.printStackTrace();}}}
}

5 测试报错NotSerializableException:

在这里插入图片描述
报错原因:要序列化对象所在的类并没有实现序列化接口
解决方案:实现序列化接口

6 测试报错InvalidClassException:

在这里插入图片描述

报错原因:本次反序列化时使用的UID与序列化时的UID不匹配
解决方案:反序列化时的UID与序列化时的UID要保持一致,或者测试时一次序列操作对应一次反序列化操作,否则不匹配就报错

在这里插入图片描述

7 为什么反序列化版本号需要与序列化版本号一致?

我们在反序列化时,JVM会拿着反序列化流中的serialVersionUID与序列化时相应的实体类中的serialVersionUID来比较,如果不一致,就无法正常反序列化,出现序列化版本不一致的异常InvalidClassException。

而且我们在定义需要序列化的实体类时,如果没有手动添加UID,
Java序列化机制会根据编译的class自动生成一个,那么只有同一次编译生成的class才是一样的UID。

如果我们手动添加了UID,只要这个值不修改,就可以不论编译次数,进行序列化和反序列化操作。

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

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

相关文章

leetcode 26.删除有序数组中的重复项

⭐️ 题目描述 🌟 leetcode链接:删除有序数组中的重复项 代码: /*思路:双指针问题[1,1,2]src-> [ 1 , 1 , 2 ]destnums[src] nums[dest] > src;src-> [ 1 , 1 , 2 ]destnums[src]…

计算机视觉:多相机硬件同步拍摄

计算机视觉:多相机硬件同步拍摄 传感器同步硬件同步信号FSYNC信号STROBE信号 硬件接线硬件设备接线步骤: 软件驱动参考文献 传感器同步 目前主要有两种方法来同步不同传感器的信息(帧、IMU数据包、ToF等): 硬件同步&…

7.用python写网络爬虫,验证码处理

前言 验证码(CAPTCHA)的全称为全自动区分计算机和人类的公开图灵测试(Completely Automated Public Turing testtotellComputersand Humans Apart)从其全称可以看出,验证码用 于测试用户是否为真实人类。一个典型的验证…

基于Web的小学学科数字教学资源管理系统

摘要 小学学科数字教学资源管理是一个典型的学习项目,从教学资源、教材信息的统计和分析,在过程中会产生大量的、各种各样的数据。本文以小学学科数字教学资源管理系统为目标,采用B/S模式,以Springboot为开发框架,java…

STM32单片机(五)第二节:EXTI外部中断练习2(旋转编码器计次)

❤️ 专栏简介:本专栏记录了从零学习单片机的过程,其中包括51单片机和STM32单片机两部分;建议先学习51单片机,其是STM32等高级单片机的基础;这样再学习STM32时才能融会贯通。 ☀️ 专栏适用人群 :适用于想要…

Git常用命令

个人学习笔记,记录已经使用过的相关命名 1. 分支管理 2. 回退提交 2.1. 相关命令 2.2. 应用 回退后可以通过git log查看当前处于哪一版本 2.3. 撤销回退 先使用git reflog,找到需要回退的版本号,例如这里撤销回退之前提交测试文件的版本…

计算机网络的89个核心概念

主机:计算机网络上任何一种能够连接网络的设备都被称为主机或者说端系统,比如手机、平板电脑、电视、游戏机、汽车等,随着 5G 的到来,将会有越来越多的终端设备接入网络。 通信链路:通信链路是由物理链路(…

App 抓包提示网络异常怎么破?

背景 当你测试App的时候,想要通过Fiddler/Charles等工具抓包看下https请求的数据情况,发现大部分的App都提示网络异常/无数据等等信息。以“贝壳找房”为例: Fiddler中看到的请求是这样的: 你可能开始找证书的问题:是…

微信小程序canvas层级太高,与其他非原生组件层级冲突

官网已经提出新版本以支持同层渲染,但是实际项目中层级还是冲突的。 最后在文档中找到这样一段话,用真机打开,层级就正常了 。所以建议大家,多使用真机调试去测试!!!!

中介者模式(二十一)

相信自己,请一定要相信自己 上一章简单介绍了观察者模式(二十), 如果没有看过, 请观看上一章 一. 中介者模式 引用 菜鸟教程里面中介者模式介绍: https://www.runoob.com/design-pattern/mediator-pattern.html 中介者模式(Mediator Pattern&#xff…

rabbitmq安装步骤和遇到的问题

一、安装准备工具 1.下载Eralng,下面链接已提供otp_win64_20.2.exe 链接: https://pan.baidu.com/s/1lmvCMPVAV1Ba9UogCdQpZg 提取码:x9m7 2.下载rabbitmq,下面链接已提供rabbitmq-server-3.7.4.exe 链接: https:…

vtkdicom0.8_vtk9.2_dcmtk3.6.7_qt6.2编译OK

目录 0 结果展示 1 cmake要点 2 编译报错解决 3 参考链接 0 结果展示 1 cmake要点 注意DCMTK_dcmtk_INCLUDE_DIR 2 编译报错解决 D:\Work\C\qt6Work\DCMTK\install\bin\dcmtkcharls.dll : fatal error LNK1107: 文件无效或损坏: 无法在 0x308 处读取 修改,从…