`
talentluke
  • 浏览: 592444 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

自定义Java的序列化与反序列化

 
阅读更多

 

要想将父类对象也序列化,就需要让父类也实现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,要有无参数的默认构造函数,重写void writeExternal(ObjectOutput out) throws IOException; 

void readExternal(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 方法对序列化内容进行处理。

 

 

参考https://www.ibm.com/developerworks/cn/java/j-lo-serial/

 http://201211131343.iteye.com/blog/1772780

 

 

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics