文章目录
- 1. ObjectInputStream 和 OjbectOutputSteam
- 2. 对象的序列化
假设有一个Person对象
- 把这个对象存到电脑的硬盘上,硬盘存储的基础是什么?(二进制)。那就需要把对象转化为一个二进制的字节流,把这个流保存到电脑上。要使用这个对象,得把流转化为对象在使用。
- 把这个对象通过网络传到另一个机器上,网络的通信基础是什么?(二进制)。也就是需要把一个对象转化为二进制的数据流,把这个流通过网络进行传输,在接收端如果要使用接收的对象得先把对象的流转化为对象。
正是因为①保存对象到硬盘(对象的持久化)和②对象的网络传输,需要做这两件事,就产生了对象的输入与输出
1. ObjectInputStream 和 OjbectOutputSteam
概念:用于存储和读取对象的处理流。它的强大之处就是可以把 Java 中的对象写入到数据源中,也能把对象从数据源中还原回来。
序列化 (Serialize) :用 ObjectOutputStream 类将一个 Java 对象写入IO 流中
反序列化 (Deserialize) :用 ObjectInputStream 类从 IO 流中恢复该 Java 对象
注意:ObjectOutputStream 和 ObjectInputStream 不能序列化 static 和 transient修饰的成员变量
序列化和反序列化针对的是对象的各种属性,不包括类的属性。
2. 对象的序列化
对象序列化机制允许把内存中的 Java 对象转换成平台无关的二进制流,从而允许把这种二进制流持久地保存在磁盘上,或通过网络将这种二进制流传输到另一个网络节点。当其它程序获取了这种二进制流,就可以恢复成原来的 Java 对象。
序列化的好处在于可将任何实现了 Serializable 接口的对象转化为字节数据,使其在保存和传输时可被还原
序列化是 RMI ( Remote Method Invoke – 远程方法调用)过程的参数和返回值都必须实现的机制,而 RMI 是 JavaEE 的基础。因此序列化机制是 JavaEE 平台的基础。
如果需要让某个对象支持序列化机制,则必须让其类是可序列化的,为了让某个类是可序列化的,该类必须实现如下两个接口之一:
①Serializable
②Externalizable
凡是实现 Serializable 接口的类都有一个表示序列化版本标识符的静态变量:
private static final long serialVersionUID;
serialVersionUID 用来表明类的不同版本间的兼容性
如果类没有显示定义这个静态变量,它的值是 Java 运行时环境根据类的内部细节自动生成的。若类的源代码作了修改,serialVersionUID 可能发生变化。故建议,显示声明。
显示定义 serialVersionUID 的用途:
希望类的不同版本对序列化兼容,因此需确保类的不同版本具有相同的 serialVersionUID
不希望类的不同版本对序列化兼容,因此需确保类的不同版本具有不同的 serialVersionUID
使用对象流序列化对象
序列化:
若某个类实现了 Serializable 接口,该类的对象就是可序列化的:
①创建一个 ObjectOutputStream
②调用 ObjectOutputStream 对象的 writeObject( 对象 ) 方法输出可序列化对象。注意写出一次,操作 flush()
反序列化:
①创建一个 ObjectInputStream
②调用 readObject() 方法读取流中的对象
强调:如果某个类的字段不是基本数据类型或 String 类型,而是另一个引用类型,那么这个引用类型必须是可序列化的,否则拥有该类型的 Field 的类也不能序列化
举例:创建一个Person对象,对其进行序列化和反序列化
Person类:
package day13;
import java.io.Serializable;
/**
* 序列化与反序列化的对象
* @author 14532
*/
public class Person implements Serializable{
/**
* 一个表示序列化版本标识符的静态变量
* 用来表明类的不同版本间的兼容性
*/
private static final long serialVersionUID = 1L;
public String name;
public int age;
}
Test.java:
(1)序列化对象
package day13;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
/**
* 序列化与反序列化
* @author 14532
*
*/
public class Test5 {
public static void main(String[] args) {
try {
Test5.testSerialize();
} catch (Exception e) {
e.printStackTrace();
}
}
/**
* 对象的序列化
*/
public static void testSerialize() throws Exception{
//定义对象的输出流,把对象的序列化之后的流放到指定的文件中
ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("D:/java_doc/eclipse-workspace/demo/src/day13/tt9.txt"));
Person p= new Person();
p.name = "zhangsan";
p.age = 11;
out.writeObject(p);
out.flush();
out.close();
}
}
运行结果:在指定路径中创建tt9.txt文件,并写入序列化后的流
(2)反序列化
package day13;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
/**
* 序列化与反序列化
* 注意:对象的序列化与反序列化使用的类要严格一致,包名、类名、类结构等等所有都要一致
* @author 14532
*
*/
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:/java_doc/eclipse-workspace/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:/java_doc/eclipse-workspace/demo/src/day13/tt9.txt"));
Object obj = in.readObject();
Person p = (Person) obj;
System.out.println(p.name);
System.out.println(p.age);
in.close();
}
}
运行结果: