【图灵干货】Java高级教程第四节:Java 序列化
2021-10-28 17:05:29
Java提供了一种对象序列化机制,在这种机制中,对象可以用一系列字节来表示,这个字节序列包括这个对象的数据、关于对象类型的信息以及存储在对象中的数据类型。
当序列化对象被写到文件后,就能从文件中读取,反序列化,即对象的类型信息,对象的数据,以及对象中的数据类型,可用于在内存中创建新对象。
所有的流程都是独立于Java虚拟机(JVM)的,也就是说,在一个平台上序列化的对象可以在另一个完全不同的平台上对它进行反序列化。
类别ObjectInputStream和ObjectOutputStream是包含反序列化和序列化对象方法的高级数据流。
ObjectOutputStream类包含许多编写不同数据类型的方法,但有一种特殊的方法例外:
publicfinalvoidwriteObject(对象)throwsIOException。
上述方法序列化了一个对象,并将其发送到输出流。类似ObjectInputStream类包含以下方法来反序列化一个对象:
publicfinalObjectreadObject()throwsIOException,ClassNotFoundException。
这个方法从流中取出下一个对象,然后反序列化对象。其返回值是Object,因此,您需要将其转换为适当的数据类型。
要说明序列化在Java中是如何工作的,我将使用前面教程中提到的Employee类,假定我们已经定义了一个Employee类,它实现了Serializable接口。
Employee.java文件代码:
publicclassEmployeimplementsjava.io{
公共stringname;
公共Stringaddress;
公开传输SSN;
公共数字;公共数字
publicvoidmailCheck{
println(“Mailingacheckto”+name+"+address);
}}
要注意,一个类的对象要成功地串行化,必须满足两个条件:
这个类必须实现java.io.Serializable对象。
这个类的所有属性必须可以序列化。当某个属性无法被序列化时,该属性必须指出它是transition。
若您希望了解Java标准类是否可以序列化,请查看该类的文档。很容易检查类的实例能否序列化,只要看一下这个类有没有实现java.io.Serializable接口。
二、对象序列化。
ObjectOutputStream类用于序列化一个对象,SerializeDemo示例实例化一个Employee对象,并将该对象序列化为一个文件。
在程序执行之后,您将创建一个employee.ser文件。这个程序不会有任何输出,但你可以通过研读代码了解这个程序。
注:当将对象序列化为文件时,为文件提供一个.ser扩展名,这符合Java的标准惯例。
SerializeDemo.java 文件代码:
import java.io.*;
public class SerializeDemo {
public static void main(String [] args) {
Employee e = new Employee();
e.name = "Reyan Ali";
e.address = "Phokka Kuan, Ambehta Peer";
e.SSN = 11122333;
e.number = 101;
try {
FileOutputStream fileOut = new FileOutputStream("/tmp/employee.ser");
ObjectOutputStream out = new ObjectOutputStream(fileOut);
out.writeObject(e);
out.close();
fileOut.close();
System.out.printf("Serialized data is saved in /tmp/employee.ser");
}
catch(IOException i) {
i.printStackTrace();
} } }
三、反序列化对象
下面的 DeserializeDemo 程序实例了反序列化,/tmp/employee.ser 存储了 Employee 对象。
DeserializeDemo.java 文件代码:
import java.io.*;
public class DeserializeDemo {
public static void main(String [] args) {
Employee e = null;
try {
FileInputStream fileIn = new FileInputStream("/tmp/employee.ser");
ObjectInputStream in = new ObjectInputStream(fileIn);
e = (Employee) in.readObject();
in.close();
fileIn.close();
}
catch(IOException i) {
i.printStackTrace();
return;
}
catch(ClassNotFoundException c) {
System.out.println("Employee class not found");
c.printStackTrace();
return;
}
System.out.println("Deserialized Employee...");
System.out.println("Name: " + e.name);
System.out.println("Address: " + e.address);
System.out.println("SSN: " + e.SSN);
System.out.println("Number: " + e.number);
} }
以上程序编译运行结果如下所示:
Deserialized Employee...
Name: Reyan Ali
Address:Phokka Kuan, Ambehta Peer
SSN: 0
Number:101
这里要注意以下要点:
readObject() 方法中的 try/catch代码块尝试捕获 ClassNotFoundException 异常。对于 JVM 可以反序列化对象,它必须是能够找到字节码的类。如果JVM在反序列化对象的过程中找不到该类,则抛出一个 ClassNotFoundException 异常。
注意,readObject() 方法的返回值被转化成 Employee 引用。
当对象被序列化时,属性 SSN 的值为 111222333,但是因为该属性是短暂的,该值没有被发送到输出流。所以反序列化后 Employee 对象的 SSN 属性为 0。
下一章节我们将介绍Java高级教程第五节:Java 网络编程
免费java架构师视频学习地址:免费视频