Java序列化和反序列化示例介绍
作者:Black·Tea 发布时间:2023-11-25 04:24:26
以前用序列化都是一些方法需要才实现的,后来业务需求要深拷贝才去研究。参阅了别人博客得出一些总结。
序列化是为了把Java对象转化为字节序列(字节流)的过程。然后深拷贝是通过对流的操作来实现的,序列化后数据方便存储和传输。反序列化则是把字节序列反序列化为Java对象
存储方便:因为对象会被回收,序列化后可以持续化存储在磁盘中
传输方便:字节序列(二进制形式)可以进行网络传输和传播。
最好设置一个SerialversionUID,因为序列化和反序列化是对比SerialversionUID来进行的,虽然不设置接口也会默认生成一个,但是要知道序列化对象过程一般都是对象->序列化->存储或传输->反序列化。
举个例子:
先创建一个实体类Student
import lombok.Data;
import java.io.Serializable;
@Data
public class Student implements Serializable {
private Integer id;
private String name;
private String sex;
}
然后创建一个测试类SerializableTest
import serialization.entity.Student;
import java.io.*;
public class SerializableTest {
public static void main(String[] args) throws Exception {
serializeStudent();
Student student = deserializeStudent();
System.out.println("name:" + student.getName());
System.out.println("sex:" + student.getSex());
}
private static void serializeStudent() throws IOException {
Student student = new Student();
student.setId(1);
student.setName("张三");
student.setSex("male");
ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream(
new File("F:/student.txt")));
out.writeObject(student);
System.out.println("序列化成功");
out.close();
}
private static Student deserializeStudent() throws Exception {
ObjectInputStream in = new ObjectInputStream(new FileInputStream(new File("F:/student.txt")));
Student student = (Student) in.readObject();
System.out.println("反序列化成功");
return student;
}
}
执行结果:
序列化成功
反序列化成功
name:张三
sex:male
这个时候没有指定SerialversionUID也是可以成功的,但对象->序列化->存储或传输->反序列化,咱们在反序列化操作之前对Student类修改呢?
这个时候咱们修改一下代码,先注释掉反序列化代码,先进行序列化。
import serialization.entity.Student;
import java.io.*;
public class SerializableTest {
public static void main(String[] args) throws Exception {
serializeStudent();
// Student student = deserializeStudent();
// System.out.println("name:" + student.getName());
// System.out.println("sex:" + student.getSex());
}
private static void serializeStudent() throws IOException {
Student student = new Student();
student.setId(1);
student.setName("张三");
student.setSex("male");
ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream(
new File("F:/student.txt")));
out.writeObject(student);
System.out.println("序列化成功");
out.close();
}
// private static Student deserializeStudent() throws Exception {
// ObjectInputStream in = new ObjectInputStream(new FileInputStream(new File("F:/student.txt")));
// Student student = (Student) in.readObject();
// System.out.println("反序列化成功");
// return student;
// }
}
运行结果:
序列化成功
修改Student类
import lombok.Data;
import java.io.Serializable;
@Data
public class Student implements Serializable {
private Integer id;
private String name;
private String sex;
private String address;
}
注释掉序列化方法,进行反序列化
import serialization.entity.Student;
import java.io.*;
public class SerializableTest {
public static void main(String[] args) throws Exception {
// serializeStudent();
Student student = deserializeStudent();
System.out.println("name:" + student.getName());
System.out.println("sex:" + student.getSex());
}
// private static void serializeStudent() throws IOException {
// Student student = new Student();
// student.setId(1);
// student.setName("张三");
// student.setSex("male");
//
// ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream(
// new File("F:/student.txt")));
// out.writeObject(student);
// System.out.println("序列化成功");
// out.close();
// }
private static Student deserializeStudent() throws Exception {
ObjectInputStream in = new ObjectInputStream(new FileInputStream(new File("F:/student.txt")));
Student student = (Student) in.readObject();
System.out.println("反序列化成功");
return student;
}
}
执行结果:
Exception in thread "main" java.io.InvalidClassException: serialization.entity.Student; local class incompatible: stream classdesc serialVersionUID = 3846952599709361171, local class serialVersionUID = -4606152942663467236
at 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)
at serialization.demo.SerializableTest.deserializeStudent(SerializableTest.java:30)
at serialization.demo.SerializableTest.main(SerializableTest.java:10)
Process finished with exit code 1
可以看出两次的执行的SerialversionUID不匹配,导致产生java.io.InvalidClassException异常,所以只要指定了SerialversionUID就不会报异常。
//指定serialVersionUID正确写法
private static final long serialVersionUID = 3846952599709361171L;
//如果已经进行序列化了不知道SerialversionUID,可以通过反射获取
Object obj = Student.class.newInstance();
Field field = Student.class.getDeclaredField("serialVersionUID");
field.setAccessible(true);
System.out.println(field.getLong(obj));
最后需要知道的一点就是字节流和字符流的区别。
字节流:传输过程中,传输数据的最基本单位是字节的流。
字符流:传输过程中,传输数据的最基本单位是字符的流。
这样讲可能有点不知所云,字节其实就是Java的八大基本类型Byte(比特)单位,而字符通常是’A’、‘B’、’$’、’&'等,字节大小则取决于你是什么编码(环境),如下:
ASCII 码中,一个英文字母(不分大小写)为一个字节,一个中文汉字为两个字节。
UTF-8 编码中,一个英文字为一个字节,一个中文为三个字节。
Unicode 编码中,一个英文为一个字节,一个中文为两个字节。
符号:英文标点为一个字节,中文标点为两个字节。例如:英文句号 . 占1个字节的大小,中文句号 。
占2个字节的大小。UTF-16 编码中,一个英文字母字符或一个汉字字符存储都需要 2 个字节(Unicode 扩展区的一些汉字存储需要 4 个字节)。
UTF-32 编码中,世界上任何字符的存储都需要 4 个字节。
来源:https://blog.csdn.net/qq_42706375/article/details/122294354


猜你喜欢
- 本文实例展示了DevExpress根据条件设置GridControl RepositoryItem是否可编辑的方法。一般在C#项目的开发中,
- equalsIgnoreCase() 方法用于将字符串与指定的对象比较,不考虑大小写。实例equals() 会判断大小写区别,equalsI
- 题目描述BM99 顺时针旋转矩阵描述 有一个NxN整数矩阵,请编写一个算法,将矩阵顺时针旋转90度。 给定一个NxN的矩阵,和矩阵的阶数N,
- 现在智能手机基本都是触摸操作,点击按钮是一种交互方式,同时手势相关的操作,比如滑动等等同样是很重要的交互方式。这篇文章是对安卓手势交互相关知
- 这篇文章主要介绍了java接口私有方法实现过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可
- 最近项目中遇到了华为虚拟按键适配的问题,主页是个RecylerView(如下图),如果不做适配,在界面初始化完毕后,虚拟按键会遮挡页面或者空
- Http请求类package wzh.Http;import java.io.BufferedReader;import java.io.I
- 前言上篇博客我们介绍了如何创建一个spring项目,并且如何的存、取对象,介绍了相关方法,那么本篇博客将接着上篇博客的内容介绍如何更加简单的
- Kotlin JSON解析开发环境就是下面这个…一、添加依赖在build.gradle.kts文件中添加下面依赖。de
- 本文实例为大家分享了C语言运用函数指针数组制作计算器的具体代码,供大家参考,具体内容如下先来回顾一下概念:指针数组 —— 存放指针的数组函数
- 一.相关知识:Java多线程程序设计到的知识:(一)对同一个数量进行操作(二)对同一个对象进行操作(三)回调方法使用(四)线程同步,死锁问题
- 理解枚举类型枚举类型是Java 5中新增特性的一部分,它是一种特殊的数据类型,之所以特殊是因为它既是一种类(class)类型却又比类类型多了
- Java本身都是值传递式的调用,对于对象传递的是地址值。给地址值重新赋值等于重新指向,不会影响外层。而且这里Integer对象也有特殊性。其
- 目录一、Eureka概述1、Eureka特点2、Eureka两大组件3、Eureka三大角色二、Eureka Server服务注册中心1、p
- 上篇文章中,我们讲了在 Spring Security 中如何踢掉前一个登录用户,或者禁止用户二次登录,通过一个简单的案例,实现了我们想要的
- 1.基本语法key: value;kv之间有空格大小写敏感使用缩进表示层级关系缩进不允许使用tab,只允许空格缩进的空格数不重要,只要相同层
- FeignClient重试机制造成的接口幂等性Feign源码分析,其实现类在 SynchronousMethodHandler,实现方法是p
- 1、如何解决服务之间的通信问题?[1]HTTP REST方式 使用http协议进行数据传递 json格式数据[2]RPC方式 远程过程调用
- 通常的函数调用 一个通常的函数调用的例子://自行包含头文件void MyFun(int x); //此
- 本文介绍的仿IOS对话框的实现,先来看一下效果图具体代码如下:public class AlertDialog { private Cont