Java之序列化SerialVersionUID的作用

转载 2016年06月01日 15:35:16

先总结下:

         Serializable接口 标记类可以被序列化(ObjectOutputStream)和反序列化(ObjectInputStream)
暴露了类的内存代表二进制数据,private的属性域不安全。默认的序列化将Object的Class name, non-tranient fields 和域值编码成二进制序列,二进制序列可以反编译成Object实例。类的名称、域名和域类型改变就会打破序列化的兼容,并且使新老版本的类兼容更复杂化。每一个序列化的类可以指派一个serialVersionUID, 默认为该类和类成员的hashcode,serialVersionUID也会被包括在序列化里面,所以反序列化可以检测version。如果默认情况(没有显示指定一个serialVersionUID)序列化的数据中serialVersionUID跟Class的serialVersioinUID不同,反序列化就失败。所以可以显示指定一个serialVersionUID,所以序列化机制就可以向前和向后兼容的(对于那些同样的serialVersionUID)。如果你不想跟之前的版本兼容,比如加入、修改和删除了一个域,你可以increment serialVersionUID的值。可以实现writeObject(ObjectOutputStream out) 和readObject(ObjectInputStream in)。不可能改变类名字而保持序列化兼容,所以匿名类不能使用序列化。可以使用transient来排除序列化的属性域,反序列化时,这些transient域会根据其类型被赋值为null、0、或false。transient只能修饰变量,不能修饰类和方法,不能修饰本地变量,静态变量不管是否被transient,均不能被序列化。在Java中,对象的序列化可以通过实现两种接口来实现,若实现的是Serializable接口,则所有的序列化将会自动进行,若实现的是Externalizable接口,则没有任何东西可以自动序列化,需要在writeExternal方法中进行手工指定所要序列化的变量。



Java序列化是将一个对象编码成一个字节流,反序列化将字节流编码转换成一个对象。 序列化是Java中实现持久化存储的一种方法;为数据传输提供了线路级对象表示法。

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

Eclipse中The serializable class XXXXXX  does not declare a static final serialVersionUID field of type long出现这样的警告处理办法。 
当采用程序的Add default Serial version ID修复时,Eclipse会加上:private static final long serialVersionUID = 1L; 
当采用程序的Add generated Serial version ID修复时,Eclipse会加上:private static final long serialVersionUID = xxxxL; 

其实这个问题出现的具体原因是和序列化中的这个serialVersionUID有关。 serialVersionUID 用来表明类的不同版本间的兼容性。有两种生成方式: 一个是默认的1L;另一种是根据类名、接口名、成员方法及属性等来生成一个64位的哈希字段 。
在JDK中,可以利用JDK的bin目录下的serialver.exe工具产生这个serialVersionUID 的值,对于Test.class,执行命令: 

serialver Test   这时JVM(java虚拟机)会生成一个哈希字段。 

对比一下这个哈希字段的值与方法2中生成的字段值是一样的,可见,在CMD中使用serialver指令就是根据类名、接口名、成员方法及属性等来生成哈希字段的。 

java类中为什么需要重载 serialVersionUID 属性。 
当两个进程在进行远程通信时,彼此可以发送各种类型的数据。无论是何种类型的数据,都会以二进制序列的形式在网络上传送。发送方需要把这个Java对象转换为字节序列,才能在网络上传送;接收方则需要把字节序列再恢复为Java对象。 把Java对象转换为字节序列的过程称为对象的序列化,把字节序列恢复为Java对象的过程称为对象的反序列化。 

  对象的序列化主要有两种用途: 
  1) 把对象的字节序列永久地保存到硬盘上,通常存放在一个文件中; 
  2) 在网络上传送对象的字节序列。 
java.io.ObjectOutputStream代表对象输出流,它的writeObject(Object obj)方法可对参数指定的obj对象进行序列化,把得到的字节序列写到一个目标输出流中。 java.io.ObjectInputStream代表对象输入流,它的readObject()方法从一个源输入流中读取字节序列,再把它们反序列化为一个对象,并将其返回。 只有实现了Serializable或Externalizable接口的类的对象才能被序列化。Externalizable接口继承自Serializable接口,实现Externalizable接口的类完全由自身来控制序列化的行为,而仅实现Serializable接口的类可以采用默认的序列化方式 。 凡是实现Serializable接口的类都有一个表示序列化版本标识符的静态变量:private static final long serialVersionUID; 类的serialVersionUID的默认值完全依赖于Java编译器的实现,对于同一个类,用不同的Java编译器编译,有可能会导致不同的serialVersionUID。显式地定义serialVersionUID有两种用途: 

  1)在某些场合,希望类的不同版本对序列化兼容,因此需要确保类的不同版本具有相同的serialVersionUID;在某些场合,不希望类的不同版本对序列化兼容,因此需要确保类的不同版本具有不同的serialVersionUID。 

  2)当你序列化了一个类实例后,希望更改一个字段或添加一个字段,不设置serialVersionUID,所做的任何更改都将导致无法反序化旧有实例,并在反序列化时抛出一个异常。如果你添加了serialVersionUID,在反序列旧有实例时,新添加或更改的字段值将设为初始化值(对象为null,基本类型为相应的初始默认值),字段被删除将不设置。 

序列化算法一般会按步骤

  1. 将对象实例相关的类元数据输出。
  2. 递归地输出类的超类描述直到不再有超类。
  3. 类元数据完了以后,开始从最顶层的超类开始输出对象实例的实际数据值。
  4. 从上至下递归输出实例的数据
////////////////////////////////////////////////

下面说说用Eclipse/Myeclipse生成serialVersionUID,方法非常简单。

    有图为例,比如访客表需要添加serialVersionUID

    以TEST.JAVA为例,先接口implements Serializable

    

    使用Eclipse/Myeclipse IDE时,会出现左侧的黄色叹号提示,点击这个黄色叹号

    选择第一项,添加已生成的串行版本标识,则Eclipse/Myeclipse IDE 会生成一个serialVersionUID,搞定。



感谢http://blog.csdn.net/linghu_java/article/details/25194821,http://www.shangxueba.com/jingyan/1856064.html



关于serialVersionUID的说明

1、为什么要使用serialVersionUID (1)对于实现了Serializable接口的类,可以将其序列化输出至磁盘文件中,同时会将其serialVersionUID输出到文件中。 (2)...
  • jediael_lu
  • jediael_lu
  • 2014年05月24日 11:02
  • 10533

serialVersionUID的作用和生成方式

serialVersionUID的作用: 1,一个类如果要用于传输或长久地存到硬盘,必须变成二进制的形式,再次用到就需要将二进制还原回来,这就是序列化于反序列化,但怎么保证中间不出错呢,那么就需要为...
  • yxb_yingu
  • yxb_yingu
  • 2016年09月19日 16:49
  • 454

Java中序列化的serialVersionUID作用

Java序列化是将一个对象编码成一个字节流,反序列化将字节流编码转换成一个对象。 序列化是Java中实现持久化存储的一种方法;为数据传输提供了线路级对象表示法。 Java的序列化机制是通过在运行...
  • z69183787
  • z69183787
  • 2015年07月10日 19:44
  • 3242

java序列化和反序列化以及serialVersionUID的作用

当两个进程在进行远程通信时,彼此可以发送各种类型的数据。无论是何种类型的数据,都会以二进制序列的形式在网络上传送。发送方需要把这个Java对象转换为字节序列,才能在网络上传送;接收方则需要把字节序列再...
  • liaoqianwen123
  • liaoqianwen123
  • 2013年12月11日 21:10
  • 687

Java对象的serialVersionUID在序列化和反序列化的用途

本博客主要转自如下链接 http://blog.csdn.net/javazejian/article/details/52665164 这篇文章写的不错,但是有些地方我估计博主没有亲自测试,所以有...
  • u011734144
  • u011734144
  • 2016年09月29日 12:44
  • 1539

Java中serialVersionUID的作用

Java中serialVersionUID的作用
  • CHZiroy
  • CHZiroy
  • 2015年05月20日 14:43
  • 2875

Java Serializable序列化,反序列化 ,serialVersionUID的作用

Java序列化是指把Java对象转换为字节序列的过程;而Java反序列化是指把字节序列恢复为Java对象的过程。如此可以把对象转换成字节流,可以进行网络传输,保持到本地文件,数据库等,增加对象的生命周...
  • chenchaofuck1
  • chenchaofuck1
  • 2016年03月07日 16:31
  • 1526

Java对象的serialVersionUID在序列化和反序列化的用途

本博客主要转自如下链接 http://blog.csdn.net/javazejian/article/details/52665164 这篇文章写的不错,但是有些地方我估计博主没有亲自测试,所以有...
  • u011734144
  • u011734144
  • 2016年09月29日 12:44
  • 1539

Java中序列化的serialVersionUID作用

Java序列化是将一个对象编码成一个字节流,反序列化将字节流编码转换成一个对象。 序列化是Java中实现持久化存储的一种方法;为数据传输提供了线路级对象表示法。 Java的序列化机制是通过在运行...
  • z69183787
  • z69183787
  • 2015年07月10日 19:44
  • 3242

序列化版本号serialVersionUID的作用_动力节点Java学院整理

  • 2017年10月30日 16:16
  • 54KB
  • 下载
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Java之序列化SerialVersionUID的作用
举报原因:
原因补充:

(最多只允许输入30个字)