serialVersionUID解析

/ 技术 / 3 条站内评论 / 2193浏览

前言


private static final long serialVersionUID = 1L;项目中经常有用到,查了相关资料写个总结。


作用


简单来说,Java的序列化机制是通过在运行时判断类的serialVersionUID来验证版本一致性的。变量serialVersionUID称为序列化版本号,这个变量多用于实现了Serializable的类中,试用场景是类的序列化。

在进行反序列化时,JVM会把传来的字节流中的serialVersionUID与本地相应实体(类)的serialVersionUID进行比较,如果相同就认为是一致的,可以进行反序列化,否则就会出现序列化版本不一致的异常。(InvalidClassException)

反序列化的时候,虚拟机同样会先读取该变量值,然后再当前读取的类中寻找同样的变量值,如果找到,那么反序列话成功,找不到即会报异常。

使用虚拟机默认计算的serialVersionUID就会有一个明显的劣势,那就是类一旦序列化后,我们就不能修改该类了,因为计算的serialVersionUID会改变,导致后期反序列话失败。

为了避免以上情况,我们手动定义一个静态常量来人为定义serialVersionUID,因为一旦手动定义了,虚拟机就不会再进行计算了,这也就是你上面写的那就话。


serialVersionUID有两种显示的生成方式:

一个是默认的1L,比如:private static final long serialVersionUID = 1L;

一个是根据类名、接口名、成员方法及属性等来生成一个64位的哈希字段,

比如:private static final long serialVersionUID = xxxxL;

----引用知乎


异常


java.io.InvalidClassException: com.test.Serial; local class incompatible: stream classdesc serialVersionUID = 7144449072310778281, local class serialVersionUID = -1172001890485818398
at java.io.ObjectStreamClass.initNonProxy(Unknown Source)
at java.io.ObjectInputStream.readNonProxyDesc(Unknown Source)
at java.io.ObjectInputStream.readClassDesc(Unknown Source)
at java.io.ObjectInputStream.readOrdinaryObject(Unknown Source)
at java.io.ObjectInputStream.readObject0(Unknown Source)
at java.io.ObjectInputStream.readObject(Unknown Source)
at com.test.serializable.DeserialTest.main(DeserialTest.java:19)


IntelliJ IDEA 中自动生成 serialVersionUID


安装插件

file>setting>plugins>搜索GenerateSerialVersionUID


重启即可


使用


快捷键(alt+insert)


或者右键选择



选择





OK

子类继承父类序列化一些问题

父类实现了Serializable,子类不需要实现Serializable

  相关注意事项
    a)序列化时,只对对象的状态进行保存,而不管对象的方法;
    b)当一个父类实现序列化,子类自动实现序列化,不需要显式实现Serializable接口;
    c)当一个对象的实例变量引用其他对象,序列化该对象时也把引用对象进行序列化
    d)并非所有的对象都可以序列化,至于为什么不可以,有很多原因了,比如:
        1.安全方面的原因,比如一个对象拥有private,public等field,对于一个要传输的对象,比如写到文件,或者进行rmi传输等等,在序列化进行传输的过程中,这个对象的private等域是不受保护的。
       2. 资源分配方面的原因,比如socket,thread类,如果可以序列化,进行传输或者保存,也无法对他们进行重新的资源分配,而且,也是没有必要这样实现。

  例子:

 1,父类实现了Serializable,子类没有,

父类有int a = 1int b = 2int c = 3

子类有int d = 4int e = 5

序列化子类的时候,d和e会不会被序列化?(答案:会)

 

2,反过来父类未实现Serializable,子类实现了,序列化子类实例的时候,父类的属性是直接被跳过不保存,还是能保存但不能还原?(答案:值不保存)

 

解:父类实现接口后,所有派生类的属性都会被序列化。子类实现接口的话,父类的属性值丢失。

java中序列化之子类继承父类序列化

当一个父类实现Serializable接口后,他的子类都将自动的实现序列化。

详细内容参考java中序列化之子类继承父类序列化

资料参考

https://www.zhihu.com/question/24852886  

http://www.cnblogs.com/youxin/archive/2013/06/04/3116304.html  

https://blog.csdn.net/qq_35246620/article/details/77686098

  1. 刚好用到,太棒了

    回复
  2. SAn

    selenIUM自动测试:serialVersionUID解析 - SAn Blog

    回复
  3. SAn

    selenIUM自动测试:serialVersionUID解析 - SAn Blog

    回复
召唤蕾姆
琼ICP备18000156号

鄂公网安备 42011502000211号