前言
对象经常要通过IO进行传送,让你写程序传递对象,你会怎么做?把对象的状态数据用某种格式写入到硬盘,Person->“zxx,male,28,30000”Person
,既然大家都要这么干,并且没有个统一的干法,于是,Sun公司
就提出一种统一的解决方案,它会把对象变成某个格式进行输入和输出,这种格式对程序员来说是透明(transparent)的,但是,我们的某个类要想能被 Sun公司
的这种方案处理,必须实现Serializable
接口。
ObjectOutputStream.writeObject(obj);
Object obj = ObjectInputStream.readObject();
假设两年前我保存了某个类的一个对象,这两年来,我修改该类,删除了某个属性和增加了另外一个属性,两年后,我又去读取那个保存的对象,或有什么结果?未知!Sun
的 jdk
就会蒙了。为此,一个解决办法就是在类中增加版本后,每一次类的属性修改,都应该把版本号升级一下,这样,在读取时,比较存储对象时的版本号与当前类的版本号,如果不一致,则直接报版本号不同的错!
总结
简单来说,Java
的序列化机制是通过在运行时判断类的serialVersionUID
来验证版本一致性的。在进行反序列化时,JVM会把传来的字节流中的serialVersionUID
与本地相应实体(类)的serialVersionUID
进行比较,如果相同就认为是一致的,可以进行反序列化,否则就会出现序列化版本不一致的异常。
当实现java.io.Serializable
接口的实体(类)没有显式地定义一个名为serialVersionUID
,类型为long
的变量时,Java序列化机制会根据编译的class自动生成一个serialVersionUID
作序列化版本比较用,这种情况下,只有同一次编译生成的class才会生成相同的serialVersionUID
。
如果我们不希望通过编译来强制划分软件版本,即实现序列化接口的实体能够兼容先前版本,未作更改的类,就需要显式地定义一个名为serialVersionUID
,类型为long
的变量,不修改这个变量值的序列化实体都可以相互进行串行化和反串行化。
后记
以上两段话摘自http://blog.sina.com.cn/s/blog_3e9d2b3501011uy8.html,以前只知道序列化ID的此作用,但实际并未碰到过。昨天有幸一见,现把场景回忆如下:
环境:
1)分布式即多台resin服务器;
2)序列化的 java 类没有显示定义serialVersionUID
;
3)使用了Redis作为缓存(序列化后的对象存入redis缓存中);
在使用过程,报序列化时候版本不兼容,即serialVersionUID
不同异常。由于不显示定义serialVersionUID
,在分布式环境,相同类的class在不同resin在的class的serialVersionUID
不同,反序列化就容易报此异常。
所以,在使用序列化过程中,最好显示定义序列号。