深入理解Java对象序列化

关于Java序列化的文章早已是汗牛充栋了,本文是对我个人过往学习,理解及应用Java序列化的一个总结。此文内容涉及Java序列化的基本原理,以及多种方法对序列化形式进行定制。

作者:Sha Jiang 来源:Sha Jiang的博客| 2012-02-14 10:29

关于Java序列化的文章早已是汗牛充栋了,本文是对我个人过往学习,理解及应用Java序列化的一个总结。此文内容涉及Java序列化的基本原理,以及多种方法对序列化形式进行定制。在撰写本文时,既参考了Thinking in Java, Effective Java,JavaWorld,developerWorks中的相关文章和其它网络资料,也加入了自己的实践经验与理解,文、码并茂,希望对大家有所帮助。(持续更新中,2012.02.13最后更新)

1. 什么是Java对象序列化

Java平台允许我们在内存中创建可复用的Java对象,但一般情况下,只有当JVM处于运行时,这些对象才可能存在,即,这些对象的生命周期不会比JVM的生命周期更长。但在现实应用中,就可能要求在JVM停止运行之后能够保存(持久化)指定的对象,并在将来重新读取被保存的对象。Java对象序列化就能够帮助我们实现该功能。

使用Java对象序列化,在保存对象时,会把其状态保存为一组字节,在未来,再将这些字节组装成对象。必须注意地是,对象序列化保存的是对象的"状态",即它的成员变量。由此可知,对象序列化不会关注类中的静态变量。

除了在持久化对象时会用到对象序列化之外,当使用RMI(远程方法调用),或在网络中传递对象时,都会用到对象序列化。Java序列化API为处理对象序列化提供了一个标准机制,该API简单易用,在本文的后续章节中将会陆续讲到。

2. 简单示例

在Java中,只要一个类实现了java.io.Serializable接口,那么它就可以被序列化。此处将创建一个可序列化的类Person,本文中的所有示例将围绕着该类或其修改版。

Gender类,是一个枚举类型,表示性别

  
  
  1. public enum Gender {  
  2.     MALE, FEMALE  

如果熟悉Java枚举类型的话,应该知道每个枚举类型都会默认继承类java.lang.Enum,而该类实现了Serializable接口,所以枚举类型对象都是默认可以被序列化的。

Person类,实现了Serializable接口,它包含三个字段:name,String类型;age,Integer类型;gender,Gender类型。另外,还重写该类的toString()方法,以方便打印Person实例中的内容。

  
  
  1. public class Person implements Serializable {  
  2.  
  3.     private String name = null;  
  4.  
  5.     private Integer age = null;  
  6.  
  7.     private Gender gender = null;  
  8.  
  9.     public Person() {  
  10.         System.out.println("none-arg constructor");  
  11.     }  
  12.  
  13.     public Person(String name, Integer age, Gender gender) {  
  14.         System.out.println("arg constructor");  
  15.         this.name = name;  
  16.         this.age = age;  
  17.         this.gender = gender;  
  18.     }  
  19.  
  20.     public String getName() {  
  21.         return name;  
  22.     }  
  23.  
  24.     public void setName(String name) {  
  25.         this.name = name;  
  26.     }  
  27.  
  28.     public Integer getAge() {  
  29.         return age;  
  30.     }  
  31.  
  32.     public void setAge(Integer age) {  
  33.         this.age = age;  
  34.     }  
  35.  
  36.     public Gender getGender() {  
  37.         return gender;  
  38.     }  
  39.  
  40.     public void setGender(Gender gender) {  
  41.         this.gender = gender;  
  42.     }  
  43.  
  44.     @Override 
  45.     public String toString() {  
  46.         return "[" + name + ", " + age + ", " + gender + "]";  
  47.     }  

SimpleSerial,是一个简单的序列化程序,它先将一个Person对象保存到文件person.out中,然后再从该文件中读出被存储的Person对象,并打印该对象。

  
  
  1. public class SimpleSerial {  
  2.  
  3.     public static void main(String[] args) throws Exception {  
  4.         File file = new File("person.out");  
  5.  
  6.         ObjectOutputStream oout = new ObjectOutputStream(new FileOutputStream(file));  
  7.         Person person = new Person("John"101, Gender.MALE);  
  8.         oout.writeObject(person);  
  9.         oout.close();  
  10.  
  11.         ObjectInputStream oin = new ObjectInputStream(new FileInputStream(file));  
  12.         Object newPerson = oin.readObject(); // 没有强制转换到Person类型  
  13.         oin.close();  
  14.         System.out.println(newPerson);  
  15.     }  

上述程序的输出的结果为:

  
  
  1. arg constructor  
  2. [John, 31, MALE] 

此时必须注意的是,当重新读取被保存的Person对象时,并没有调用Person的任何构造器,看起来就像是直接使用字节将Person对象还原出来的。

当Person对象被保存到person.out文件中之后,我们可以在其它地方去读取该文件以还原对象,但必须确保该读取程序的CLASSPATH中包含有Person.class(哪怕在读取Person对象时并没有显示地使用Person类,如上例所示),否则会抛出ClassNotFoundException。

3. Serializable的作用

为什么一个类实现了Serializable接口,它就可以被序列化呢?在上节的示例中,使用ObjectOutputStream来持久化对象,在该类中有如下代码:

  
  
  1. private void writeObject0(Object obj, boolean unshared) throws IOException {  
  2.       ...
  3.     if (obj instanceof String) {  
  4.         writeString((String) obj, unshared);  
  5.     } else if (cl.isArray()) {  
  6.         writeArray(obj, desc, unshared);  
  7.     } else if (obj instanceof Enum) {  
  8.         writeEnum((Enum) obj, desc, unshared);  
  9.     } else if (obj instanceof Serializable) {  
  10.         writeOrdinaryObject(obj, desc, unshared);  
  11.     } else {  
  12.         if (extendedDebugInfo) {  
  13.             throw new NotSerializableException(cl.getName() + "\n" 
  14.                     + debugInfoStack.toString());  
  15.         } else {  
  16.             throw new NotSerializableException(cl.getName());  
  17.         }  
  18.     }  
  19.     ...  

从上述代码可知,如果被写对象的类型是String,或数组,或Enum,或Serializable,那么就可以对该对象进行序列化,否则将抛出NotSerializableException。

4. 默认序列化机制

如果仅仅只是让某个类实现Serializable接口,而没有其它任何处理的话,则就是使用默认序列化机制。使用默认机制,在序列化对象时,不仅会序列化当前对象本身,还会对该对象引用的其它对象也进行序列化,同样地,这些其它对象引用的另外对象也将被序列化,以此类推。所以,如果一个对象包含的成员变量是容器类对象,而这些容器所含有的元素也是容器类对象,那么这个序列化的过程就会较复杂,开销也较大。

5. 影响序列化

在现实应用中,有些时候不能使用默认序列化机制。比如,希望在序列化过程中忽略掉敏感数据,或者简化序列化过程。下面将介绍若干影响序列化的方法。

5.1 transient关键字

当某个字段被声明为transient后,默认序列化机制就会忽略该字段。此处将Person类中的age字段声明为transient,如下所示,

  
  
  1. public class Person implements Serializable {  
  2.     ...  
  3.     transient private Integer age = null;  
  4.     ...  

再执行SimpleSerial应用程序,会有如下输出:

  
  
  1. arg constructor  
  2. [John, null, MALE] 

可见,age字段未被序列化。

5.2 writeObject()方法与readObject()方法

对于上述已被声明为transitive的字段age,除了将transitive关键字去掉之外,是否还有其它方法能使它再次可被序列化?方法之一就是在Person类中添加两个方法:writeObject()与readObject(),如下所示:

  
  
  1. <li class="alt" style="list-style-type: decimal; border: none; line-height: 21px; font-family: Arial; background: url("http://images.51cto.com/images/art1105/images/0.gif") -498px -70px repeat-y scroll transparent; color: inherit; padding: 0px 3px 0px 10px !important; margin: 0px !impor
本项目是一个基于SSM(Spring+SpringMVC+MyBatis)框架和Vue.js前端技术的大学生第二课堂系统,旨在为大学生提供一个便捷、高效的学习和实践平台。项目包含了完整的数据库设计、后端Java代码实现以及前端Vue.js页面展示,适合计算机相关专业的毕设学生和需要进行项目实战练习的Java学习者。 在功能方面,系统主要实现了以下几个模块:用户管理、课程管理、活动管理、成绩管理和通知公告。用户管理模块支持学生和教师的注册、登录及权限管理;课程管理模块允许教师上传课程资料、设置课程时间,并由学生进行选课;活动管理模块提供了活动发布、报名和签到功能,鼓励学生参与课外实践活动;成绩管理模块则用于记录和查询学生的课程成绩和活动参与情况;通知公告模块则实时发布学校或班级的最新通知和公告。 技术实现上,后端采用SSM框架进行开发,Spring负责业务逻辑层,SpringMVC处理Web请求,MyBatis进行数据库操作,确保了系统的稳定性和扩展性。前端则使用Vue.js框架,结合Axios进行数据请求,实现了前后端分离,提升了用户体验和开发效率。 该项目不仅提供了完整的源代码和相关文档,还包括了详细的数据库设计文档和项目部署指南,为学习和实践提供了便利。对于基础较好的学习者,可以根据自己的需求在此基础上进行功能扩展和优化,进一步提升自己的技术水平和项目实战能力。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值