其实这个问题简单思考一下就可以搞清楚,方法是不带状态的,就是一些指令,指令是不需要序列化的,只要你的JVM classloader可以load到这个类,那么类方法指令自然就可以获得。序列化真正需要保存的只是对象属性的值,和对象的类型。
这些知识找一本Java基础编程的书,或者Java手册就可以查到,我以为是不应该犯这种基本概念错误的。 我们可以做一个简单的小试验,来证实一下:
package com.javaeye; import java.io.Serializable; public class DomainObject implements Serializable { private String name; private int age ; public int getAge(); { return age; } public void setAge(int age); { this.age = age; } public String getName(); { return name; } public void setName(String name); { this.name = name; } }
package com.javaeye; import java.io.FileOutputStream; import java.io.ObjectOutputStream; public class Main { public static void main(String[] args); throws Exception { DomainObject obj = new DomainObject();; obj.setAge(29);; obj.setName("fankai");; FileOutputStream fos = new FileOutputStream("DomainObject");; ObjectOutputStream oos = new ObjectOutputStream(fos);; oos.writeObject(obj);; oos.close();; fos.close();; } } DomainObject是我们准备序列化的类,在Main里面,我们new一个DomainObject的对象,然后赋值,最后把该对象序列化到一个硬盘文件中。 然后使用一种支持二进制编辑器,例如UltraEdit打开这个文件,看看Java都对DomainObject序列化了哪些信息,你就什么都明白了。 为了更方便观察,我使用Linux下面的strings去提取文本信息,输出为: robbin@linux:~> strings DomainObject com.javaeye.DomainObject ageL namet Ljava/lang/String;xp fankai 这些信息很直观的告诉我们序列化都保存了些什么内容: 1)对象的类型 2)对象属性的类型 3)对象属性的值 并没有什么方法签名的信息,更不要说什么序列化方法了。 然后我们再做一个试验,给DomainObject增加两个方法:
package com.javaeye; import java.io.Serializable; public class DomainObject implements Serializable { private String name; private int age ; public int getAge(); { return age; } public void setAge(int age); { this.age = age; } public String getName(); { return name; } public void setName(String name); { this.name = name; } public String toString(); { return "This is a serializable test!"; } public void doSomeWork(); { System.out.println("hello");; } } 我们增加了toString方法和doSomeWork方法,按照你的理论,如果序列化方法的话,产生的文件体积必然增大。记录一下文件体积,92Byte,好了,删除,运行程序,生成了新的文件,看一下体积,还是92Byte! 拿到Linux下面再提取一下字符串: robbin@linux:~> strings DomainObject com.javaeye.DomainObject ageL namet Ljava/lang/String;xp fankai 完全一模一样! 然后我们再做第三个试验,这次把DomainObject的两个属性以及相关方法删除掉:
package com.javaeye; import java.io.Serializable; public class DomainObject implements Serializable { public String toString(); { return "This is a serializable test!"; } public void doSomeWork(); { System.out.println("hello");; } } 修改Main类如下:
package com.javaeye; import java.io.FileOutputStream; import java.io.ObjectOutputStream; public class Main { public static void main(String[] args); throws Exception { DomainObject obj = new DomainObject();; FileOutputStream fos = new FileOutputStream("DomainObject");; ObjectOutputStream oos = new ObjectOutputStream(fos);; oos.writeObject(obj);; oos.close();; fos.close();; } }
补充: 序列化在 Effective Java 中讲得很清楚啊, 一般认为只声明实现 implements 接口, 不提供自定义的序列化形式是不负责任的做法, 这样可能导致比较多的问题比如类的不同版本之间的兼容性, 看看 Effective Java 中的条目吧 谨慎地实现 Serialiable |