序列化知识点总结

1.应用场景

JVM间数据通信解决方案

2.如何声明一个可序列化的类?

  • 一个类可序列化的前提:实现Serializable接口
  • 一个属性不想序列化的方式:增加transient关键字

3.serialVersionUID

serialVersionUID 是表示一个类的序列化标识,和一个类的类名以及包路径一起组成类在序列化时的唯一标识。
当两个jvm中存在serialVersionUID+类名+包路径一致的类时,这两个jvm可以通过序列化达到此类数据共享的目的。
当设置serialVersionUID为1L时,指代忽略serialVersionUID 的校验。
image.png
当原来类里面声明的serialVersionUID为1L,再把文件里面的serialVersionUID改为2时,会报如下错误:
java.io.InvalidClassException: demo2.Employee; local class incompatible: stream classdesc serialVersionUID = 2, local class serialVersionUID = 1

如果想深入了解serialVersionUID方面知识,可参考:https://www.cnblogs.com/duanxz/p/3511695.html

4.transient 与 static、transient 与 final

  • static属性存储在方法区,不参与序列化过程。
  • final 属性直接通过值参与序列化过程,加transient不会生效。

static属性存储在方法区,不参与序列化过程。static属性表达的并不是一个对象的状态,所以也不应该参与序列化过程。所以在效果上看与transient类似,但是在机制上有本质的区别。
image.png

5.序列化机制

  • 序列化时,如果对一个对象多次序列化,只有第一次保存对象状态,之后都会保存该对象的引用。
  • 序列化时,如果父类没有实现序列化接口,则父类状态不会序列化。
  • 反序列化时,并不调用构造函数。
  • 反序列化时,如果父类没有实现序列化接口,则会调用父类的无参构造函数生成一个父类对象。

PS:这里再说一下创建对象的几种方式:

(1) 用new语句创建对象,这是最常见的创建对象的方法。
(2) 运用反射手段,调用java.lang.Class或者java.lang.reflect.Constructor类的newInstance()实例方法。
(3) 调用对象的clone()方法。
(4) 运用反序列化手段,调用java.io.ObjectInputStream对象的 readObject()方法。
(1)和(2)都会明确的显式的调用构造函数 ;(3)是在内存上对已有对象的影印,所以不会调用构造函数 ;(4)是从文件中还原类的对象,也不会调用构造函数。

6.定制序列化规则

出于隐私保护,逻辑扩展,保持单例特性等原因,可能会产生定制序列化规则的需求,可以实现的方案有三种:

  • 自定义writeObject,readObject方法。
  • 实现Externalizable的writeExternal和readExternal方法。
  • 自定义writeReplace,readResolve方法。

三种方案的优先级顺序:

  • 序列化时 writeReplace > writeExternal > writeObject
  • 反序列化时 readResolve > readExternal > readObject
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Apple_Web

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值