码迷,mamicode.com
首页 > 其他好文 > 详细

序列化接口的id的作用

时间:2017-02-12 01:06:06      阅读:217      评论:0      收藏:0      [点我收藏+]

标签:base   环境   异常   没有   strong   作用   lang   定义   tput   

对象经常要通过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不同,反序列化就容易报此异常。

    所以,在使用序列化过程中,最好显示定义序列号。

序列化接口的id的作用

标签:base   环境   异常   没有   strong   作用   lang   定义   tput   

原文地址:http://www.cnblogs.com/MessiAndDream/p/6390336.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!