Java序列化

2021-07-04  本文已影响0人  云芈山人

为什么设计了Java序列化的功能?

Java平台允许我们在内存中创建可复用的Java对象,但一般情况下,只有当JVM运行时,这些对象才存在,即这些对象的生命周期不会大于JVM的生命周期。但现实应用中,就可能要求JVM在停止运行后依然能保存(持久化)指定的对象【保存(持久化)对象及其状态到内存或磁盘中】,并在将来重新读取被保存的对象。Java对象序列化能够帮助我们实现该功能。

除了在持久化对象时会用到对象序列化之外,当使用RMI(远程方法调用),或在网络中传递对象时,都会用到对象序列化。Java序列化API为处理对象序列化提供了一个标准机制,该API简单易用。

Java提供了一种对象序列化的机制。该机制中,一个对象可以被表示为一个字节序列,该字节序列包括该对象的数据、有关对象的类型的信息和存储在对象中的数据类型。
将序列化对象写入文件之后,可以从文件中读取出来,并且对它进行反序列化,也可以说,对象的类型信息、对象的数据、对象的数据类型可以在内存中新建对象。

注意:使用Java对象序列化,在保存对象时,会把其状态保存为一组字节,在未来,再将这些字节组装成对象。对象序列化保存的是对象的“状态”,即它的成员变量。由此可知,Java序列化不保存静态变量!

类ObjectInputStream() 和ObjectOutputStream()是最高层次的数据流,它们包含反序列化和序列化的功能。

如何实现序列化和反序列化

实现序列化方法

1. 实现Serializable接口
/*
 * @author  unascribed
 * @see java.io.ObjectOutputStream
 * @see java.io.ObjectInputStream
 * @see java.io.ObjectOutput
 * @see java.io.ObjectInput
 * @see java.io.Externalizable
 * @since   JDK1.1
 */
public interface Serializable {
}

优点:内建支持,易于实现。
缺点:占用空间过大,由于额外的开销导致速度变比较慢 。

2. 实现Externalizable 方法(外部化)

自己对要序列化的内容进行控制,控制哪些属性能被序列化,哪些不能被序列化。

public interface Externalizable extends java.io.Serializable {
     void writeExternal(ObjectOutput out) throws IOException;
     void readExternal(ObjectInput in) throws IOException, ClassNotFoundException;
}

优点:开销较少(程序员决定存储什么),可能的速度提升。
缺点:虚拟机不提供任何帮助,也就是说所有的工作都落到了开发人员的肩上。

3. Serializable 与 Externalizable的区别

反序列化

虚拟机是否允许反序列化,不仅取决于类路径和功能代码是否一致,一个非常重要的一点是两个类的序列化 ID 是否一致(就是 private static final long serialVersionUID)。

  1. 实现Serializable接口的对象在反序列化时不需要调用对象所在类的构造方法,完全基于字节。
  2. 实现Externalizable接口的方法在反序列化时会调用构造方法。

常见的序列化协议

COM

主要用于Windows平台,并没有真正跨平台,另外COM的序列化原理利用了编译器中虚表,使得其学习成本巨大。

CORBA

是早期比较好的实现跨平台、跨语言的序列化协议。但它的主要问题是参与方过多带来的版本过多,兼容性较差,使用复杂晦涩。

XML&SOAP

JSON(JavaScript Object Notation)

Thrift

Facebook开源提供的一个高性能,轻量级RPC服务框架,其产生正是为了满足当前大数据量、分布式、跨语言、跨平台数据通讯的需求。Thrift在空间开销和解析性能上有了比较大的提升,对于性能要求比价高的分布式系统,它是一个优秀的RPC解决方案;但是由于Thrift的序列化被嵌入到Thrift框架里面,Thrift框架本身并没有透出序列化和反序列化的接口,这导致其很难和其他传输层协议共同使用。

上一篇下一篇

猜你喜欢

热点阅读