【13】java 序列化ID的作用

       序列化ID的作用:  

       序列化ID决定:是否能够成功反序列化!简单来说,java的序列化机制是通过在运行时判断类的serialVersionUID来验证版本一致性的。在进行反序列化时,JVM会把传来的字节流中的serialVersionUID与本地实体类中的serialVersionUID进行比较,如果相同则认为是一致的,便可以进行反序列化,否则就会报序列化版本不一致的异常。等会我们可以通过代码验证一下。

       序列化ID如何产生:

       当我们一个实体类中没有显示的定义一个名为“serialVersionUID”、类型为long的变量时,Java序列化机制会根据编译时的class自动生成一个serialVersionUID作为序列化版本比较,这种情况下,只有同一次编译生成的class才会生成相同的serialVersionUID。譬如,当我们编写一个类时,随着时间的推移,我们因为需求改动,需要在本地类中添加其他的字段,这个时候再反序列化时便会出现serialVersionUID不一致,导致反序列化失败。那么如何解决呢?便是在本地类中添加一个“serialVersionUID”变量,值保持不变,便可以进行序列化和反序列化。

       验证“serialVersionUID”不一致导致反序列化失败

[java] view plain copy  print?

  1. import java.io.Serializable;  
  2.   
  3. /** 
  4.  *  
  5.  * 测试序列化和反序列化 
  6.  * @author crazyandcoder 
  7.  * @date [2015-8-5 上午11:14:32] 
  8.  */  
  9. public class Person implements Serializable  {  
  10.       
  11.     private int age;  
  12. //  private String sex;  
  13. //  private String name;  
  14. //  private String hobby;  
  15.     //序列化ID  
  16. //  private static final long serialVersionUID = -5809782578272943999L;  
  17.       
  18. //  public String getHobby() {  
  19. //      return hobby;  
  20. //  }  
  21. //  
  22. //  public void setHobby(String hobby) {  
  23. //      this.hobby = hobby;  
  24. //  }  
  25.   
  26.     public Person() {}  
  27.       
  28.     public int getAge() {  
  29.         return age;  
  30.     }  
  31.       
  32.     public void setAge(int age) {  
  33.         this.age = age;  
  34.     }  
  35.       
  36. //  public String getSex() {  
  37. //      return sex;  
  38. //  }  
  39. //  
  40. //  public void setSex(String sex) {  
  41. //      this.sex = sex;  
  42. //  }  
  43.       
  44. //  public String getName() {  
  45. //      return name;  
  46. //  }  
  47. //  public void setName(String name) {  
  48. //      this.name = name;  
  49. //  }  
  50.   
  51. }  

        复用前篇使用到的代码,首先,我们生成一个本地Person类,里面添加一个字段age,然后将其序列化存于本地E:/hello.txt中,

[java] view plain copy  print?

  1. import java.io.FileNotFoundException;  
  2. import java.io.FileOutputStream;  
  3. import java.io.IOException;  
  4. import java.io.ObjectOutputStream;  
  5.   
  6.   
  7. /** 
  8.  *  
  9.  * 测试序列化和反序列化 
  10.  * @author crazyandcoder 
  11.  * @date [2015-8-5 上午11:16:14] 
  12.  */  
  13. public class ObjSerializeAndDeserializeTest {  
  14.   
  15.        
  16.     public static void main(String[] args) {  
  17.           
  18.         //将Person对象序列化  
  19.         SerializePerson();  
  20.     }  
  21.       
  22.       
  23.     /** 
  24.      *  
  25.      * @author crazyandcoder 
  26.      * @Title: 序列化Person对象,将其存储到 E:/hello.txt文件中 
  27.      * @param   
  28.      * @return void  
  29.      * @throws  
  30.      * @date [2015-8-5 上午11:21:27] 
  31.      */  
  32.     private static void SerializePerson() {  
  33.         Person person =new Person();  
  34.         person.setAge(30);  
  35.         ObjectOutputStream outputStream = null;  
  36.         try {  
  37.             outputStream=new ObjectOutputStream(new FileOutputStream("E:/hello.txt"));  
  38.             outputStream.writeObject(person);  
  39.             System.out.println("序列化成功。");  
  40.         } catch (FileNotFoundException e) {  
  41.             e.printStackTrace();  
  42.                   
  43.         } catch (IOException e) {  
  44.             e.printStackTrace();  
  45.                   
  46.         } finally {  
  47.             try {  
  48.                 outputStream.close();  
  49.             } catch (IOException e) {  
  50.                 e.printStackTrace();  
  51.             }  
  52.         }  
  53.           
  54.     }  
  55. }  

        运行一下,会在控制台中打印“序列化成功。”,然后我们在Person类中再添加一个字段,name,然后直接从E:/hello.txt中反序列化,再运行一下,看看会出现什么问题。

[java] view plain copy  print?

  1. import java.io.FileInputStream;  
  2. import java.io.FileNotFoundException;  
  3. import java.io.FileOutputStream;  
  4. import java.io.IOException;  
  5. import java.io.ObjectInputStream;  
  6. import java.io.ObjectOutputStream;  
  7.   
  8. /** 
  9.  *  
  10.  * 测试序列化和反序列化 
  11.  *  
  12.  * @author crazyandcoder 
  13.  * @date [2015-8-5 上午11:16:14] 
  14.  */  
  15. public class ObjSerializeAndDeserializeTest {  
  16.   
  17.     public static void main(String[] args) {  
  18.   
  19.         // 反序列化生成Person对象  
  20.         Person person = DeserializePerson();  
  21.         System.out.println("name :" + person.getName());  
  22.         System.out.println("age  :" + person.getAge());  
  23.     }  
  24.   
  25.     /** 
  26.      * 执行反序列化过程生产Person对象 
  27.      *  
  28.      * @author crazyandcoder 
  29.      * @Title: DeserializePerson 
  30.      * @param @return 
  31.      * @return Person 
  32.      * @throws 
  33.      * @date [2015-8-5 下午1:30:12] 
  34.      */  
  35.     private static Person DeserializePerson() {  
  36.   
  37.         Person person = null;  
  38.         ObjectInputStream inputStream = null;  
  39.         try {  
  40.             inputStream = new ObjectInputStream(new FileInputStream("E:/hello.txt"));  
  41.             try {  
  42.                 person = (Person) inputStream.readObject();  
  43.                 System.out.println("执行反序列化过程成功。");  
  44.             } catch (ClassNotFoundException e) {  
  45.                 e.printStackTrace();  
  46.             }  
  47.         } catch (FileNotFoundException e) {  
  48.             e.printStackTrace();  
  49.         } catch (IOException e) {  
  50.             e.printStackTrace();  
  51.         } finally {  
  52.             try {  
  53.                 inputStream.close();  
  54.             } catch (IOException e) {  
  55.                 e.printStackTrace();  
  56.             }  
  57.         }  
  58.         return person;  
  59.     }  
  60. }  


       运行一下,不出意外,报了一个异常。

 

        从上面两张图便可以看出两次的序列化ID是不一样的,导致反序列化失败。

总结:

       虚拟机是否允许反序列化,不仅取决于类路径和功能代码是否一致,一个非常重要的一点是两个类的序列化 ID 是否一致(就是 private static final long serialVersionUID = 1L)。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java高级程序设计 第5章 Java序列化机制 5.1 应用场景 5.2 相关知识5.3 实施过程 5.4 拓展知识5.5 拓展训练 5.6 课后小结5.7 课后习题 5.8 上机实训 Java高级程序设计实战教程第五章-Java序列化机制全文共15页,当前为第1页。 5.1 应用场景 在分布式环境下,当进行远程通信时,彼此可以发送各种类型的数据。无论是何种类型的数据,都会以二进制序列的形式在网络上传输。发送方需要把这个Java对象转换为字节序列,才能在网络上传送;接收方则需要把字节序列再恢复为Java对象。 序列化是一种将对象以一连串的字节描述的过程,用于解决在对对象流进行读写操作时所引发的问题。序列化可以将对象的状态写在流里进行网络传输,或者保存到文件、数据库等系统中,并在需要时把该流读取出来重新构造一个相同的对象。 Java高级程序设计实战教程第五章-Java序列化机制全文共15页,当前为第2页。 5.2 相关知识 5.2.1 序列化的概念 5.2.2 序列化应用 5.2.3 序列化的几种方式 5.2.4 对象实现机制 Java高级程序设计实战教程第五章-Java序列化机制全文共15页,当前为第3页。 5.2.1 序列化的概念 将在内存中的各种对象的状态(也就是实例变量,不是方法)保存在磁盘中或者在网络中进行传输,并且可以把保存的对象状态再读出来。 将一个Java对象写入IO流;与此对应的,则是从IO流中恢复一个Java对象。 Java提供这种保存对象状态的机制,就是序列化。 对象序列化Java编程中的必备武器 Java高级程序设计实战教程第五章-Java序列化机制全文共15页,当前为第4页。 5.2.2 序列化应用 当你想把内存中的对象状态保存到一个文件中或者数据库中时候; 当你想用套接字在网络上传送对象的时候; 当你想通过RMI传输对象的时候。 Java高级程序设计实战教程第五章-Java序列化机制全文共15页,当前为第5页。 5.2.3 序列化的几种方式 在Java中socket传输数据时,数据类型往往比较难选择。可能要考虑带宽、跨语言、版本的兼容等问题。比较常见的做法有两种: 一是把对象包装成JSON字符串传输, 二是采用Java对象的序列化和反序列化。 随着Google工具protoBuf的开源,protobuf也是个不错的选择。 *提示:对JSON,Object Serialize,ProtoBuf 做个对比。 Java高级程序设计实战教程第五章-Java序列化机制全文共15页,当前为第6页。 5.2.4 对象实现机制 为了方便开发人员将Java对象进行序列化及反序列化Java提供了一套方便的API来支持。其中包括以下接口和类: java.io.Serializable java.io.Externalizable ObjectOutput ObjectInput ObjectOutputStream ObjectInputStream Java高级程序设计实战教程第五章-Java序列化机制全文共15页,当前为第7页。 5.2 相关知识 5.2.1 序列化的概念 5.2.2 序列化应用 5.2.3 序列化的几种方式 5.2.4 对象实现机制 Java高级程序设计实战教程第五章-Java序列化机制全文共15页,当前为第8页。 5.3 实施过程 5.3.1 任务一 使用serializable序列化实体对象 5.3.2 任务二 反序列化将Person对象从磁盘上读出 Java高级程序设计实战教程第五章-Java序列化机制全文共15页,当前为第9页。 5.3.1 任务一 使用serializable序列化实体对象 实现Serializable接口非常简单,只要让Jjava实现Serializable接口即可,无需实现任何方法。 一个类一旦实现了Serializable接口,那么该类的对象就是可序列化的。实现类的对象的序列化可以使用ObjectOutputStream,实现步骤如下: 创建ObjectOutputStream对象; 调用ObjectOutputStream的writeObject方法输出对象。 Java高级程序设计实战教程第五章-Java序列化机制全文共15页,当前为第10页。 5.3.2 任务二 反序列化将Person对象从磁盘上读出 任务需求: 反序列化将Employee对象从磁盘上读出并修改员工信息,然后再写入到文件中。 分析: 相应的反序列化需要使用的类是ObjectInputStream,反序列化步骤如下: 创建ObjectInputStream对象; 使用ObjectInputStream的readObject方法取出对象。 Java高级程序设计实战教程第五章-Java序列化机制全文

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值