Java序列化与反序列化及serialVersionUID

2023-05-16

Java序列化与反序列化是什么?为什么需要序列化与反序列化?如何实现Java序列化与反序列化?本文围绕这些问题进行了探讨。

1.Java序列化与反序列化

Java序列化是指把Java对象转换为字节序列的过程;而Java反序列化是指把字节序列恢复为Java对象的过程。

2.为什么需要序列化与反序列化

我们知道,当两个进程进行远程通信时,可以相互发送各种类型的数据,包括文本、图片、音频、视频等, 而这些数据都会以二进制序列的形式在网络上传送。那么当两个Java进程进行通信时,能否实现进程间的对象传送呢?答案是可以的。如何做到呢?这就需要Java序列化与反序列化了。换句话说,一方面,发送方需要把这个Java对象转换为字节序列,然后在网络上传送;另一方面,接收方需要从字节序列中恢复出Java对象。

当我们明晰了为什么需要Java序列化和反序列化后,我们很自然地会想Java序列化的好处。其好处一是实现了数据的持久化,通过序列化可以把数据永久地保存到硬盘上(通常存放在文件里),二是,利用序列化实现远程通信,即在网络上传送对象的字节序列。

3.如何实现Java序列化与反序列化

1)JDK类库中序列化API

java.io.ObjectOutputStream:表示对象输出流

它的writeObject(Object obj)方法可以对参数指定的obj对象进行序列化,把得到的字节序列写到一个目标输出流中。

java.io.ObjectInputStream:表示对象输入流

它的readObject()方法源输入流中读取字节序列,再把它们反序列化成为一个对象,并将其返回。

在网络传输Java对象、将Java对象存储到文件、将Java对象以BLOB形式存储到数据库中时,需要对Java对象进行序列化及反序列化,标准模式是实现Serializable接口。
实现上述接口时,需要提供一个Serial Version UID,该UID用于标识类的版本。一个对象被序列化后,只要其版本不变,都可以进行反序列化,一旦
改变造成版本不一致,会抛出InvalidClassException异常。
建议显示定义UID,如果不显示定义,JVM会自动产生一个值,这个值和编译器的实现有关,不稳定,可能在不同JVM环境下出现反序列化抛出InvalidClassException异常的情况。
在Eclipse中,提供两种方式显示定义UID,一种是“add default serial version ID”,默认值为1L;另一种是“add generated serial version ID”,默认值是一个很大的数,是根据
类的具体属性而生成,当类属性有变动时,该值会更改。
建议采用第一种自动生成方法,当对类进行了不兼容性修改时,需要修改UID。
采用第二种方法时,如果修改了属性,不重新生成UID时,默认值是不会变的,也可以正常反序列化,但不推荐,毕竟UID的值与实际不符。
对类进行兼容性和不兼容性修改的情况请参见以下url:http://docs.oracle.com/javase/7/docs/platform/serialization/spec/version.html。
Hibernate的pojo类建议也采用上述方法,便于扩展。
对于继承关系,父类实现序列化接口,子类可以继承接口的实现,但需显示定义UID,因为父类UID类型为private static,不可被继承,同时子类作为单独的类需要单独的UID。

2)实现序列化的要求

只有实现了Serializable或Externalizable接口的类的对象才能被序列化,否则抛出异常。

3)实现Java对象序列化与反序列化的方法

假定一个Student类,它的对象需要序列化,可以有如下三种方法:

方法一:若Student类仅仅实现了Serializable接口,则可以按照以下方式进行序列化和反序列化

ObjectOutputStream采用默认的序列化方式,对Student对象的非transient的实例变量进行序列化。

ObjcetInputStream采用默认的反序列化方式,对对Student对象的非transient的实例变量进行反序列化。

方法二:若Student类仅仅实现了Serializable接口,并且还定义了readObject(ObjectInputStream in)和writeObject(ObjectOutputSteam out),则采用以下方式进行序列化与反序列化。

ObjectOutputStream调用Student对象的writeObject(ObjectOutputStream out)的方法进行序列化。

ObjectInputStream会调用Student对象的readObject(ObjectInputStream in)的方法进行反序列化。

方法三:若Student类实现了Externalnalizable接口,且Student类必须实现readExternal(ObjectInput in)和writeExternal(ObjectOutput out)方法,则按照以下方式进行序列化与反序列化。

ObjectOutputStream调用Student对象的writeExternal(ObjectOutput out))的方法进行序列化。

ObjectInputStream会调用Student对象的readExternal(ObjectInput in)的方法进行反序列化。

4)JDK类库中序列化的步骤

步骤一:创建一个对象输出流,它可以包装一个其它类型的目标输出流,如文件输出流:

ObjectOutputStream out = new ObjectOutputStream(new fileOutputStream(“D:\objectfile.obj”));

步骤二:通过对象输出流的writeObject()方法写对象:

out.writeObject(“Hello”);

out.writeObject(new Date());

5)JDK类库中反序列化的步骤

步骤一:创建一个对象输入流,它可以包装一个其它类型输入流,如文件输入流:

ObjectInputStream in = new ObjectInputStream(new fileInputStream(“D:\objectfile.obj”));

步骤二:通过对象输出流的readObject()方法读取对象:

String obj1 = (String)in.readObject();

Date obj2 = (Date)in.readObject();

说明:为了正确读取数据,完成反序列化,必须保证向对象输出流写对象的顺序与从对象输入流中读对象的顺序一致。

为了更好地理解Java序列化与反序列化,选择方法一编码实现。

Student类定义如下:

import java.io.Serializable;


public class Student implements Serializable  
 
{  


private String name;  
 private char sex;  
 private int year;  
 private double gpa;  
 public Student()  
 {  
  
 }  
 public Student(String name,char sex,int year,double gpa)  
 {  
  this.name = name;  
  this.sex = sex;  
  this.year = year;  
  this.gpa = gpa;  
 }
public String getName() {
	return name;
}
public void setName(String name) {
	this.name = name;
}
public char getSex() {
	return sex;
}
public void setSex(char sex) {
	this.sex = sex;
}
public int getYear() {
	return year;
}
public void setYear(int year) {
	this.year = year;
}
public double getGpa() {
	return gpa;
}
public void setGpa(double gpa) {
	this.gpa = gpa;
}  
 
}


序列化函数WriteObject.java

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;


public class WriteObject {

	/**
	 * @param args
	 * @throws IOException 
	 * @throws ClassNotFoundException 
	 */
	public static void writeObject() throws IOException, ClassNotFoundException {
		// TODO Auto-generated method stub

		Student st=new Student("Owen",'m',22,3.5);
		File file =new File ("d:\\test\\student.txt");
		try{
			file.createNewFile();
			
		}
		catch(IOException e){
			e.printStackTrace();
		}
		try{
			FileOutputStream fos =new FileOutputStream(file);
			ObjectOutputStream oos=new ObjectOutputStream(fos);
			oos.writeObject(st);
			oos.flush();
			oos.close();
			fos.close();
		}catch(IOException e){
			e.printStackTrace();
		}
		
		
	}

}

反序列化函数ReadObject.java

import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.ObjectInputStream;


public class ReadObject {
	public static void readObject() throws IOException, ClassNotFoundException {
		File file =new File ("d:\\test\\student.txt");
		FileInputStream fis=new FileInputStream(file);
		ObjectInputStream ois =new ObjectInputStream(fis);
		Student student = (Student) ois.readObject();
		 System.out.println("name = " + student.getName());  
		   System.out.println("sex = " + student.getSex());  
		   System.out.println("year = " + student.getYear());  
		   System.out.println("gpa = " + student.getGpa());  
		   ois.close();  
		   fis.close();  
	}
	
}

调用函数Main.java

import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.util.Date;
import java.util.Scanner;


public class Main {

	public static void main(String[] args ) throws FileNotFoundException, IOException, ClassNotFoundException{
	WriteObject.writeObject();
	System.out.println("序列化完毕");
    Scanner sc=new Scanner(System.in);
	sc.next();
	System.out.println("反序列化");
	ReadObject.readObject();
	

	}
}

结果如图
这里写图片描述

这里写图片描述

现在我们已经成功实现对象序列化与反序列化但有一个问题就是如果此时我们在生成序列化文件后,修改了Java对象,再重新反序列化,那么此时会报错。
如图:
Student类中新加一个game字段
新加game字段

会因为序列号不一致报错

怎么解决这个问题呢?答案是新加serialVersionUID
serialVersionUID: 字面意思上是序列化的版本号,这个在刚刚接触java编程时,学序列化大家一般都不会注意到,在你一个类序列化后除非你强制去掉了eclipse中warning的功能,在你实现序列化的类上会有这个警告,点击会出现增加这个版本号。
说说这个版本号得作用:就是确保了不同版本之间的兼容性,不仅能够向前兼容,还能够向后兼容,即在版本升级时反序列化仍保持对象的唯一性。
它有两种生成方式:
一个是默认的1L,比如:private static final long serialVersionUID = 1L;
一个是根据类名、接口名、成员方法及属性等来生成一个64位的哈希字段,比如: private static final long serialVersionUID = xxxxL;
从两个例子上来说明这个序列化号的作用:
这是一个类实现了序列化,但是并没有显式的声明序列号,在这里说明一下,如果没有显式声明序列号,那么在程序编译时会自己生成这个版本序列号,上述过程中就是由于加game字段之前和加game字段会生成2个不同的版本序列号,导致出现了异常。
现在为Student类加入序列号
加序列号
这里写图片描述
此时再次序列化,此时是有game字段的
这里写图片描述
这里写图片描述
删除game字段,再次执行反序列化
这里写图片描述
结果依然正确
这里写图片描述
此时把serialVersionUID删除,再次反序列化


此时如我们预料的一样,结果抛出异常
因为序列化时的serialVersionUID与此时删除game字段后的serialVersionUID是不同的,反序列化时由于2次值不同,抛出异常。

参考王路情http://blog.csdn.net/wangloveall/article/details/7992448/和
lakersuperstar http://blog.sina.com.cn/s/blog_7f73e06d0100u52c.html后自己实现了一遍,希望对大家带来帮助。
欢迎指正

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Java序列化与反序列化及serialVersionUID 的相关文章

随机推荐

  • 电子硬件3.杜邦线

    杜邦线是常用于电路连接的导线 xff0c 能够刚好插在常用的2 54mm间距的排针上 杜邦线的三种类型为 xff1a 公公线 公母线 母母线
  • Android应用安全检测工具简介

    Android应用安全检测工具简介 1 测试工具集 Appie 轻量级的软件包 可以用来进行基于Android的渗透测试 不想使用VM的时候可以尝试一下 Android Tamer 可以实时监控的虚拟环境 可以用来进行一系列的安全测试 恶意
  • C语言浮点型数据存储结构

    1 float类型 float类型占四个字节 xff0c 每个字节占8位 xff0c 总共32位 xff0c 其内存结构如下图 xff1a 31位为符号位 xff1a 0表示正数 xff0c 1表示负数 31 23位 xff1a 共8位表示
  • sockaddr和sockaddr_in详解

    struct sockaddr和struct sockaddr in这两个结构体用来处理网络通信的地址 一 sockaddr sockaddr在头文件 include lt sys socket h gt 中定义 xff0c sockadd
  • python requests模拟登陆带验证码的网站

    作为之前专利爬虫的续篇 xff0c 本篇准备描述如何通过python的requests模块登录专利查询网站 环境准备 python 3 6requests chrome尝试 首先 xff0c 我们使用chrome尝试登录专利网站 xff0c
  • 兔子与兔子(下一篇解释原理:字符串哈希)

    文章目录 兔子与兔子题目描述解题思路 兔子与兔子 题目描述 很久很久以前 xff0c 森林里住着一群兔子 有一天 xff0c 兔子们想要研究自己的 DNA 序列 我们首先选取一个好长好长的 DNA 序列 xff08 小兔子是外星生物 xff
  • 机器学习(二)--- KNN(K-Nearest Neighbors)

    KNN K Nearest Neighbors 简单类比 xff08 Simple Analogy xff09 KNN xff1a 通过你周围的人来判断你是哪一类人 Tell me about your friends who your n
  • 机器学习(三) --- DT(Decision Tree)

    文章目录 Decision TreeIntroductionConstructing Decision Treesexample Pruning决策树 随机森林和Gradient BoostingReference xff1a Decisi
  • 3D点云资料(笔记自用)

    数据集 xff1a Pointnet Pointnet2 pytorch Pointnet语义分割任务S3DIS数据集上的注意点 dataset ShapeNet简介及认识 文件格式 PCD格式 目标检测标注工具 标注工具windows安装
  • 用指针给数组赋值的一个小问题

    问题是这样的 xff0c 我打算输入与元素为数组赋值 xff0c 然后反向输出数组元素 include lt stdio h gt int main void int a 10 p 61 a i for i 61 0 i lt 10 i 4
  • 51之中断

    51单片机的中断系统 选用的单片机型号是 xff1a HC6800 ES V2 0 一 中断的概念 CPU在处理某一事件A时 xff0c 发生了另一事件B请求CPU迅速去处理 xff08 中断发生 xff09 xff1b CPU暂时中断当前
  • 基于51单片机和L298N的小车制作(一)

    到目前为止 xff0c 总算把51单片机过了一遍 xff0c 本来暑假就要完成的任务一直拖到了现在 51学完 xff0c 就想着先做点东西 xff0c 就从小车开始 玩小车最重要的是什么 xff1f 电机 这个就不用多解释了 从今天早上开始
  • 如何把android设备中的固件dump出来

    android固件是在mtdblock中 但是会有很多个block root 64 android cat proc partitions major minor blocks name 31 0 4096 mtdblock0 31 1 1
  • PID控制电机转速

    转一个PID控制电机的小程序 xff0c 被PID困扰好多天了 xff0c 知道它的原理但是一直不明白如何将它运用到电机调速中间去 xff0c 看了这个程序之后感觉茅塞顿开 原来也并不难 转载地址 xff1a 呃 xff0c 刚刚不小心把网
  • vs2017常用快捷键

    VS2017常用快快捷键 目录 43 VS常用快捷键搜集窗口操作快捷键项目快捷键调试快捷键编辑快捷键代码快捷键单元测试 xfeff xfeff 项目相关的快捷键 Ctrl 43 Shift 43 B 61 生成项目 Ctrl 43 Alt
  • phpStudy基本用法

    闲扯正文 更改目录更改首页更改端口号DNS解析 参考资料 闲扯 第一次接触php xff0c 要先安装环境 xff0c 包括Apache mySQL和PHP xff1b 安装就安装呗 xff0c 我先跑到Apache官网上面想下一个Apac
  • 小白学爬虫(三)-- requests库之Cookie

    前言使用超时参数在requests添加Cookie参数 啥是cookie三种Cookie请求方式 第一种 xff1a cookie放在headers中第二种 xff1a cookie字典传给cookies参数第三种 先发送post请求 xf
  • 操作系统(五) -- CPU的调度策略

    文章目录 常见的调度策略以及考虑的内容 CPU调度的直观想法 xff1a FIFOPriority xff08 优先级 xff09 如何评价一个调度算法的好坏 xff1f 上述三个指标之间是不是存在冲突呢 xff1f 前台任务和后台任务的关
  • vscode 前端最佳插件配置

    vscode最佳配置 最近更新时间 xff1a 2023 02 18 xff08 Vscode v1 17 xff09 vscode 配置文件内容在最后 xff0c 可直接copy使用 配置详解 editor是针对 vscode 的风格设置
  • Java序列化与反序列化及serialVersionUID

    Java序列化与反序列化是什么 xff1f 为什么需要序列化与反序列化 xff1f 如何实现Java序列化与反序列化 xff1f 本文围绕这些问题进行了探讨 1 Java序列化与反序列化 Java序列化是指把Java对象转换为字节序列的过程