序列化与反序列化
对象专属流(序列化):ObjectInputStream(反序列化)/ObjectOutputStream(序列化)
序列化(Serialize):将java对象分成一块一块编号放到硬盘文件中objectOutputStream.writeObject(对象)
反序列化(DeSerialize):从硬盘恢复成对象objectInputStream.readObject()
序列化多个对象,可将对象放到集合中,对集合对象进行序列化。只能存一个对象,存第二个对象会报错
- transient 关键字
不希望某个属性被序列化,使用transient关键字,表示游离的
private transient String name;
- 序列化版本号
版本号一致,jvm虚拟机会当成同一个类,否则反序列化十年前的文件会失败,如果版本号一致不会失败;
IDEA自动生成:private static final long serialVersionUID = 1021561711034486523L;
- IO 和 properties联合使用
properties是一个Map集合,key和value都只能是String类型
属性配置文件properties:文件内容格式为 key = value
,java规范中建议以.properties结尾
FileReader reader = new FileReader("filePath");
Properties pro = new Properties();
pro.load(reader);
Java 序列化
Java 提供了一种对象序列化的机制,该机制中,一个对象可以被表示为一个字节序列,该字节序列包括该对象的数据、有关对象的类型的信息和存储在对象中数据的类型。
将序列化对象写入文件之后,可以从文件中读取出来,并且对它进行反序列化,也就是说,对象的类型信息、对象的数据,还有对象中的数据类型可以用来在内存中新建对象。
整个过程都是 Java 虚拟机(JVM)独立的,也就是说,在一个平台上序列化的对象可以在另一个完全不同的平台上反序列化该对象。
类 ObjectInputStream 和 ObjectOutputStream 是高层次的数据流,它们包含反序列化和序列化对象的方法。
ObjectOutputStream 类包含很多写方法来写各种数据类型,但是一个特别的方法例外:
public final void writeObject(Object x) throws IOException
上面的方法序列化一个对象,并将它发送到输出流。相似的 ObjectInputStream 类包含如下反序列化一个对象的方法:
public final Object readObject() throws IOException,
ClassNotFoundException
该方法从流中取出下一个对象,并将对象反序列化。它的返回值为Object,因此,你需要将它转换成合适的数据类型。
假设我们定义了如下的Employee类,该类实现了Serializable 接口。
public class Employee implements java.io.Serializable
{
public String name;
public String address;
public transient int SSN;
public int number;
public void mailCheck()
{
System.out.println("Mailing a check to " + name
+ " " + address);
}
}
请注意,一个类的对象要想序列化成功,必须满足两个条件:
- 该类必须实现 java.io.Serializable 接口。
- 该类的所有属性必须是可序列化的。如果有一个属性不是可序列化的,则该属性必须注明是短暂的。
如果你想知道一个 Java 标准类是否是可序列化的,请查看该类的文档。检验一个类的实例是否能序列化十分简单, 只需要查看该类有没有实现 java.io.Serializable接口。
序列化对象
ObjectOutputStream 类用来序列化一个对象,如下的 SerializeDemo 例子实例化了一个 Employee 对象,并将该对象序列化到一个文件中。
该程序执行后,就创建了一个名为 employee.ser 文件。该程序没有任何输出,但是你可以通过代码研读来理解程序的作用。
注意: 当序列化一个对象到文件时, 按照 Java 的标准约定是给文件一个 .ser 扩展名。
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 对象。
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);
}
}
这里要注意以下要点:
- readObject() 方法中的 try/catch代码块尝试捕获 ClassNotFoundException 异常。对于 JVM 可以反序列化对象,它必须是能够找到字节码的类。如果JVM在反序列化对象的过程中找不到该类,则抛出一个 ClassNotFoundException 异常。
- 注意,readObject() 方法的返回值被转化成 Employee 引用。
- 当对象被序列化时,属性 SSN 的值为 111222333,但是因为该属性是短暂的,该值没有被发送到输出流。所以反序列化后 Employee 对象的 SSN 属性为 0。
也可在内存中操作,序列化为字节流
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
{
//写入字节流
ByteArrayOutputStream outByte = new ByteArrayOutputStream();
ObjectOutputStream out = new ObjectOutputStream(outByte);
out.writeObject(e);
out.close();
//分配内存,写入原始对象,生成新对象
ByteArrayInputStream ios = new ByteArrayInputStream(outByte.toByteArray());
ObjectInputStream ois = new ObjectInputStream(ios);
//返回生成的新对象
Employee employee = (Employee) ois.readObject();
System.out.println("Deserialized Employee...");
System.out.println("Name: " + employee.name);
System.out.println("Address: " + employee.address);
System.out.println("SSN: " + employee.SSN);
System.out.println("Number: " + employee.number);
outByte.close();
}catch(Exception i)
{
i.printStackTrace();
}
}
}
其它说明
- 对象属性是一个类,但属性类没有实现Serializable,对象可以序列化吗?
经过实践证明:对象属性必须实现Serializable才可以序列化,否则会抛出 java.io.NotSerializableException 异常
- 对象类继承父类,父类没有实现Serializable,对象可以序列化吗?
经过实践证明:父类没实现Serializable,对象可以序列化,但是父类的属性不能写入序列化对象中,取出时值为null;
如果想要父类的属性也写入到序列化中,则父类也需要是可序列化的,即实现Serializable接口。