要想将父类对象也序列化,就需要让父类也实现Serializable 接口。如果父类不实现的话的,就需要有默认的无参的构造函数。在父类没有实现 Serializable 接口时,虚拟机是不会序列化父对象的,而一个 Java 对象的构造必须先有父对象,才有子对象,反序列化也不例外。所以反序列化时,为了构造父对象,只能调用父类的无参构造函数作为默认的父对象。

如果子类实现了java.io.Serializable,但父类没有实现此接口,子类的值域保留下来了,但是父类的值域丢失了,这对jvm 来说是正常的,因为父类不可序列化。为了解决这个问题,只能自定义序列化行为,在序列化的默认动作之后将父类里值域写入流或写出流,顺序要一致。

自定义序列化有两种:

1.实现java.io.Serializable,重写private void writeObject(ObjectOutputStream os) throws IOException

private void readObject(ObjectInputStream is) throws IOException, ClassNotFoundException ,执行默认序列化外(先调用outputStream.defaultWriteObject();或inputStream.defaultReadObject();),可以控制声明为static或transient的数据成员。

2.实现java.io.Externalizable,要有无参数的默认构造函数,重写voidwriteExternal(ObjectOutput out) throws IOException;

voidreadExternal(ObjectInput in) throws IOException, ClassNotFoundException;完全负责序列化和恢复数据成员,除了头以外,根本没有自动序列化。实现了这个接口就不会调用1中的两个方法。

当进行序列化的时候:

首先JVM会先调用writeReplace方法,在这个阶段,我们可以进行张冠李戴,将需要进行序列化的对象换成我们指定的对象.

跟着JVM将调用writeObject方法,来将对象中的属性一个个进行序列化,我们可以在这个方法中控制住哪些属性需要序列化.

当反序列化的时候:

JVM会调用readObject方法,将我们刚刚在writeObject方法序列化好的属性,反序列化回来.

然后在readResolve方法中,我们也可以指定JVM返回我们特定的对象(不是刚刚序列化回来的对象).

注意到在writeReplace和readResolve,我们可以严格控制singleton的对象,在同一个JVM中完完全全只有唯一的对象,控制不让singleton对象产生副本.

在此writeExternal 和readExternal 的作用与writeObject和readObject 一样,当我们同时实现了两个interface的时候,JVM只运行Externalizable 接口里面的writeExternal 和readExternal 方法对序列化内容进行处理。