序列化和反序列化

对象的序列化

  1. 对象序列化机制允许把内存中的Java对象转换成平台无关的二进制流,从而允许把这种二进制流持久地保存在磁盘上,或通过网络将这种二进制流传输到另一个网络节点。当其它程序获取了这种二进制流,就可以恢复成原来的Java对象

序列化的好处在于可将任何实现了Serializable接口的对象转化为字节数据,使其在保存和传输时可被还原

序列化是 RMI(Remote Method Invoke – 远程方法调用)过程的参数和返回值都必须实现的机制,而 RMI 是 JavaEE 的基础。因此序列化机制是 JavaEE 平台的基础

如果需要让某个对象支持序列化机制,则必须让其类是可序列化的,为了让某个类是可序列化的,该类必须实现如下两个接口之一:

Serializable

Externalizable

  1. 凡是实现Serializable接口的类都有一个表示序列化版本标识符的静态变量:

    private static final long serialVersionUID;

    serialVersionUID用来表明类的不同版本间的兼容性

    如果类没有显示定义这个静态变量,它的值是Java运行时环境根据类的内部细节自动生成的。若类的源代码作了修改,serialVersionUID 可能发生变化。故建议,显示声明

显示定义serialVersionUID的用途

希望类的不同版本对序列化兼容,因此需确保类的不同版本具有相同的serialVersionUID

不希望类的不同版本对序列化兼容,因此需确保类的不同版本具有不同的serialVersionUID

实现

使用对象流序列化对象

若某个类实现了 Serializable 接口,该类的对象就是可序列化的:

创建一个 ObjectOutputStream

调用 ObjectOutputStream 对象的 writeObject(对象) 方法输出可序列化对象。注意写出一次,操作

flush()

反序列化

创建一个 ObjectInputStream

调用 readObject() 方法读取流中的对象

强调:如果某个类的字段不是基本数据类型或 String 类型,而是另一个引用类型,那么这个引用类型必须是可序列化的,否则拥有该类型的 Field 的类也不能序列化

代码示例

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;

/**
* 序列化与反序列化
* 注意:对象的序列化与反序列化使用的类要严格一致,包名,类名,类机构等等所有都要一致
* @author FadeAway
*
*/
public class Test5 {
public static void main(String[] args) {
try {
// Test5.testSerialize();
Test5.testDeserialize();
} catch (Exception e) {
e.printStackTrace();
}
}

/**
* 对象的序列化
*/
public static void testSerialize() throws Exception{
//定义对象的输出流,把对象的序列化之后的流放到指定的文件中
ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("D:/testdemo/demo/src/day13/tt9.txt"));

Person p = new Person();
p.name = "zhangsan";
p.age = 11;

out.writeObject(p);
out.flush();//刷写数据到硬盘

out.close();
}

/**
* 对象的反序列化
*/
public static void testDeserialize() throws Exception{
//创建对象输入流对象,从指定的文件中把对象序列化后的流读取出来
ObjectInputStream in = new ObjectInputStream(new FileInputStream("D:/testdemo/demo/src/day13/tt9.txt"));

Object obj = in.readObject();

Person p = (Person)obj;

//这个时候序列化与反序列化使用的类不是一个类分别是day13.Person和day13.test.Person
//这个时候反序列化就有异常day13.Person cannot be cast to day13.test.Person
// day13.test.Person p = (day13.test.Person)obj;

System.out.println(p.name);
System.out.println(p.age);

in.close();

}
}