阅读480 返回首页    go 阿里云 go 技术社区[云栖]


Java的序列化和反序列化

1.序列化是干什么的?

简单说就是为了保存在内存中的各种对象的状态,

也就是实例变量,不是方法,

并且可以把保存的对象状态再读出来。

虽然你可以用你自己的各种各样的方法来保存object states,

但是Java给你提供一种应该比你自己好的保存对象状态的机制,那就是序列化。

2.什么情况下需要序列化

当你想把的内存中的对象状态保存到一个文件中或者数据库中时候; 
当你想用套接字在网络上传送对象的时候; 
当你想通过RMI传输对象的时候; 
3.一个例子

序列化需要实现Serializable或者Externalizable 接口

下面的是实现Serializable接口,最后会提到Externalizable

Java的序列化和反序列化

Java的序列化和反序列化

Java的序列化和反序列化

如果你想学习java可以来这个群,首先是二二零,中间是一四二,最后是九零六,里面有大量的学习资料可以下载。

4.使用transient

在一些特殊场景下,比如银行账户对象,出于保密考虑,

不希望对存款金额进行序列化。或者类的一些引用类型的成员是不可序列化的。

此时可以使用transient关键字修饰不想被或者不能被序列化的成员变量。

需要注意的是transient只能修饰属性(filed),不能修饰类或方法。

一个静态变量不管是否被transient修饰,均不能被序列化。

5.自定义序列化

transient提供了一种简洁的方式将被transient修饰的成员属性完全隔离在序列化机制之外。

这样子固然不错,但是Java还提供了一种自定义序列化机制让开发者更自由地控制如何序列化各个成员属性,

或者不序列化某些属性(与transient效果相同)。

5.1 writeObject和readObject

Java的序列化和反序列化

这两个方法和ObjectOutputStream及ObjectInputStream里对应的方法名称相同。

实际上,尽管这两个方法是private型的,但是仍然是在被序列化(或反序列化)阶段被外部类ObjectOutputStream(或ObjectInputStream)调用。

仅以序列化为例:

ObjectOutputStream在执行自己的writeObject方法前会先通过反射在要被序列化的对象的类中

查找有无自定义的writeObject方法,

如有的话,则会优先调用自定义的writeObject方法。

因为查找反射方法时使用的是getPrivateMethod,

所以自定以的writeObject方法的作用域要被设置为private。

通过自定义writeObject和readObject方法可以完全控制对象的序列化与反序列化。

Java的序列化和反序列化

Java的序列化和反序列化

Java的序列化和反序列化

5.2 writeReplace和readResolve

writeReplace和readResolve是一种更彻底的序列化的机制,

它甚至可以将序列化的目标对象替换为其它的对象。

但是与writeObject和readObject不同的是,

这二者不是必须要一起使用的,而且尽量应分开使用。

若一起使用的话,只有writeReplace会生效。

6.使用Externalizable

一开始有提到过实现Externalizable接口也可以实现类的序列化。

使用这种方法,可以由开发者完全决定如何序列化和反序列化目标对象。

Externalizable接口提供了writeExternal和readExternal两个方法。

实际上这种方法和前面的自定义序列化方法很相似,

只是Externalizable强制自定义序列化。

在使用了Externalizable的类中仍可以使用writeReplace和readResolve方法。

使用Externalizable进行序列化较之使用Serializable性能略好,但是复杂度较高。

最后更新:2017-04-24 21:32:59

  上一篇:go 关于阿里的开源战略,这里有你想知道的一切
  下一篇:go 饿了么Redis Cluster集群化演进