java对象序列化反序列化_java对象的序列化以及反序列化详解

一、概念

序列化:把创建出来的对象(new出来的对象),以及对象中的成员变量的数据转化为字节数据,写到流中,然后存储到硬盘的文件中。

反序列化:可以把序列化后的对象(硬盘上的文件中的对象数据),读取到内存中,然后就可以直接使用对象。这样做的好处是不用再一次创建对象了,直接反序列化就可以了。

使用场景:

在创建对象并给所创建的对象赋予了值后,当前创建出来的对象是存放在堆内存中的,当JVM停止后,堆中的对象也被释放了,如果下一次想要继续使用之前的对象,需要再次创建对象并赋值。然而使用序列化对象,就可以把创建出来的对象及对象中数据存放到硬盘的文件中,下次使用的时候不用在重新赋值,而是直接读取使用即可。

对象直接转换为字节的形式进行网络传输

二、相关API介绍

序列化

序列化对象所属的类是ObjectOutputStream,如下图所示:

20201117161924365k9a4dzk0hudp20m_7.png

说明:ObjectOutputStream类可以把对象及其数据写入到流中或网络中。

构造函数如下图所示:

20201117161924365k9a4dzk0hudp20m_5.png

写出功能:

20201117161924365k9a4dzk0hudp20m_6.png

反序列化

反序列化对象所属的类是ObjectInputStream类:

20201117161924365k9a4dzk0hudp20m_1.png

说明:序列化输出流对象和反序列化输出流对象都不具备读写能力,分别依赖FileOutputStream和FileInputStream类来进行读写文件。

构造函数如下图所示:

20201117161924365k9a4dzk0hudp20m_10.png

读取功能:

20201117161924365k9a4dzk0hudp20m_8.png

三、实战

序列化

需求:把Student类创建的对象持久化保存。

分析和步骤:

1)自定义一个Student类,并定义name和age属性;

2)定义一个测试类,在这个测试类中创建Student类的对象s;

3)创建序列化对象objectOutputStream,同时创建输出流并关联硬盘上的文件;

4)使用序列化对象objectOutputStream调用writeObject()函数持久化保存学生对象s;

5)释放序列化对象流的资源;

Student类如下:

需要实现序列化接口Serializable,它是一个标记性接口。这个接口中没有任何的方法,这种接口称为标记型接口!它仅仅是一个标识。只有具备了这个接口标识的类才能通过Java中的序列化和反序列化流操作这个对象。

20201117161924365k9a4dzk0hudp20m_2.png

/**

* 为了保证学生对象可以被序列化,我们让Student类来实现Serializable接口

*/

public class Student implements Serializable {

private String name;

private Integer age;

//省略Getter and Setter、toString和构造方法

}

测试类如下:

/**

* 将学生对象序列化持久到硬盘文件中

*/

public class ObjectOutputStreamDemo {

public static void main(String[] args) throws IOException {

//创建学生对象

Student student = new Student("张三", 13);

//把创建出来的学生对象持久化保存在硬盘中

//创建序列化对象 创建输出流对象并关联目标文件

ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream("D:\\student.txt"));

//使用序列化对象中的方法持久化学生对象

objectOutputStream.writeObject(student);

//关闭资源

objectOutputStream.close();

}

}

序列化成功后在会在D盘生成一个student.txt文件。

注意:序列化对象如果没有实现序列化接口Serializable,则会抛出如下异常

20201117161924365k9a4dzk0hudp20m_9.png

反序列化

需求:把硬盘文件中的序列化的对象,再进行反序列操作。

分析和步骤:

1)创建反序列化对象,指定一个字节输入流,关联硬盘上的文件;

2)使用反序列对象调用函数readObject()函数,进行反序列操作,获得Student类的对象s;

3)使用对象s调用函数来获取Student类的name和age属性;

4)释放资源;

测试类如下:

/**

* 演示反序列化操作

*/

public class ObjectInputStreamDemo {

public static void main(String[] args) throws IOException, ClassNotFoundException {

//创建反序列化对象,指定一个字节输入流用来读取文件

ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream("D:\\student.txt"));

//使用反序列化对象调用函数来进行读取数据

Student student = (Student) objectInputStream.readObject();

System.out.println(student.getName() + "," + student.getAge());

//关闭资源

objectInputStream.close();

}

}

结果:

张三,13

这里如果我们对Student类做了一些简单的修改,无关紧要的修改。例如给Student类添加一个属性字段或者函数都可以,再次反序列化,就出问题了,报如下图所示的异常:

20201117161924365k9a4dzk0hudp20m_0.png

20201117161924365k9a4dzk0hudp20m_11.png

问题:什么是该类的序列版本号呢?

在序列化时将类的各个方面(如类的成员变量、成员函数、修饰符、函数返回值类型等)计算成为该类的默认 serialVersionUID值(版本号)。然后在序列化的时候,这个版本号会随着对象一起被序列化到本地文件中。serialVersionUID 称为序列版本编号(标记值)。类要进行序列化操作时,需要实现Serializable接口(Serializable接口也称为标记接口),实现了标记接口的类,该类会存在一个标记值。在反序列化的时候,从流中也就是硬盘文件中读取数据及原来存储的版本号。同时,再次根据类的内容计算当前的版本号。然后这两个版本号进行对比,如果不一致,认为类发生了改变,则抛出InvalidClassException异常。上述代码如果在反序列化之前修改Student类时,会在报的异常中出现如下图所示的提示信息:

local class incompatible: stream classdesc serialVersionUID = 8434462772094727595, local class serialVersionUID = 2117687252335805572

stream classdesc serialVersionUID = 8434462772094727595, 表示从流中读取的版本号(反序列化时)

local class serialVersionUID = 2117687252335805572 ,表示Student类的序列版本号(序列化时)

上述两个版本号不一致,所以报异常。

通过以上分析,可以得到一个结论:

如果可以保证反序列化对象和序列化对象的标记值相同,就可以避免异常的发生。

那么我们如何做才能保证反序列化对象和序列化对象的标记值相同呢?

我们修改Student类是无关紧要的。在我们修改Student类的时候,我们不希望它抛异常。我们可以给类定义一个默认的版本号,即给Student类添加标记值也就是版本号serialVersionUID。这样一来,添加的标记值即版本号会随着对象的序列化持久保存。无论是序列化,还是反序列化,都不会再根据类的各个方面计算版本号了。序列化和反序列化的版本号会永远一致,所以不会抛出异常,这样就可以避免InvalidClassException异常的发生了。

但是,这样一来,类的安全问题,只能自己来维护。因为已经将类的对象序列化之后,由于类中已经显示定义了版本号,那么反序列化的时候即使修改了Student类,也不会报异常了。

使用idea给自定义类Student添加版本号方法如下图所示:

20201117161924365k9a4dzk0hudp20m_3.png

注意:

在使用序列化操作时,不是所有的成员都可以进行序列化操作:

1)静态成员不会进行序列化操作;

2)瞬态成员也不会进行序列化操作;

瞬态成员:在进行序列化操作时,如果希望某些成员不被序列化,而该成员又不能是静态成员(不希望随着类加载而存在,和对象有关系),就使用关键字transient把成员变为瞬态成员。

代码如下所示:

20201117161924365k9a4dzk0hudp20m_4.png

反序列化的结果:

null,null

说明:由于name和age属性分别被静态和瞬态修饰了,所以都不能被序列化到硬盘上,所以反序列化都是默认值。

记住:

1、当一个对象需要被序列化 或 反序列化的时候对象所属的类需要实现Serializable接口。

2、被序列化的类中需要添加一个serialVersionUID。

序列化的细节:

序列化的时候,只能把对象在堆中的所有数据持久保存到持久设备上。静态的成员变量不会被序列化。

有时我们在序列化的时候某些非静态成员变量也不想被序列化的时候,我们可以使用瞬态关键字(transient)修饰。

四、serialVersionUID的取值

serialVersionUID的取值是Java运行时环境根据类的内部细节自动生成的。如果对类的源代码作了修改,再重新编译,新生成的类文件的serialVersionUID的取值有可能也会发生变化。

类的serialVersionUID的默认值完全依赖于Java编译器的实现,对于同一个类,用不同的Java编译器编译,有可能会导致不同的 serialVersionUID,也有可能相同。为了提高serialVersionUID的独立性和确定性,强烈建议在一个可序列化类中显示的定义serialVersionUID,为它赋予明确的值。

显式地定义serialVersionUID有两种用途:

在某些场合,希望类的不同版本对序列化兼容,因此需要确保类的不同版本具有相同的serialVersionUID;

在某些场合,不希望类的不同版本对序列化兼容,因此需要确保类的不同版本具有不同的serialVersionUID。

五、什么时候需要序列化对象

正常来说是很少需要用到的,大部分情况下我们都是使用json或者xml格式来进行数据传输。

所以一般来说如果你的对象需要网络传输或者持久化(对象直接转换为字节的形式传输),只要你的对象需要转换为字节的形式那么你的对象就要实现Serializable接口。比如使用dubbo使用rpc的方式调用接口,那么接口参数就一定要实现Serializable接口。

如果只是转换为字符串的形式与网络打交道,那么就不需要实现Serializable接口。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值