java的序列化和反序列化
参考文档
含义及使用场景
- 含义
- 序列化:把对象转换为字节序列的过程称为对象的序列化
- 反序列化:把字节序列恢复为对象的过程称为对象的反序列化
- 使用场景
- 1.当你想把的内存中的对象写入到硬盘的时候
- 2.当你想用套接字在网络上传送对象的时候
- 3.当你想通过RMI传输对象的时候
使用方法
- private static final long serialVersionUID = 1L;(不想兼容版本升级后可改为2L…)
- 根据类名、接口名、成员方法及属性等来生成一个64位的哈希字段(不想兼容添加属性后重新生成)
- 实现接口不显示的写出来 Java编译时会给你赋值(jvm版本不同可能会导致不一致)
测试
- 不实现接口 则报NotSerializableException
- 添加属性 加一个属性会引起 serialVersionUID 的变化报错 InvalidClassException
问题
类没序列化也可持久化到db中?
并不是将整个对象持久化到数据库中,而是将对象中的属性持久化到数据库中, 而这些属性(Integer/String…)都实现了Serializable接口
前后端交互不序列化也可以传输
jackson将对象解析成json的入口是ObjectMapper#writeValueAsString,其内部实现大概是先初始化一个4000容量的WriterBasedJsonGenerator#_outputBuffer char数组,
然后将对象中的字段一一往char数组中放入,最后将char数组转换成String
而String本身已经实现了Serializable接口
总结
- implements java.io.Serializable
- 虚拟机是否允许反序列化,不仅取决于类路径和功能代码是否一致,一个非常重要的一点是两个类的序列化 ID 是否一致
- 序列化并不保存静态变量
- 继承不能继承 serialVersionUID 需要子类声明
- 通过ObjectOutputStream和ObjectInputStream对对象进行序列化及反序列化
- Transient 关键字的作用是控制变量的序列化,在变量声明前加上该关键字,可以阻止该变量被序列化到文件中,在被反序列化后,transient 变量的值被设为初始值,如 int 型的是 0,对象型的是 null