答案:Java中通过实现Serializable接口实现对象序列化与反序列化,使用ObjectOutputStream和ObjectInputStream将对象转为字节流存储或传输,并需注意serialVersionUID一致性、transient字段处理、父类序列化规则及单例破坏等问题,适用于RMI、Session保存等场景,但存在安全与性能开销。

在Java中实现对象序列化与反序列化,主要是通过将对象转换为字节流,以便存储到文件或在网络中传输,之后再从字节流恢复成原始对象。这个过程称为序列化(Serialization)和反序列化(Deserialization)。以下是实际操作中的关键点和经验总结。
启用对象序列化的前提条件
要让一个类的对象支持序列化,必须实现 java.io.Serializable 接口。这是一个标记接口,不需要实现任何方法。
public class Person implements Serializable {private String name;
private int age;
// 构造方法、getter、setter 省略
}
只要类实现了 Serializable 接口,其所有基本数据类型字段和引用可序列化对象的字段都会自动被序列化。
使用 ObjectOutputStream 和 ObjectInputStream 进行序列化与反序列化
序列化对象通常写入文件,反序列化则从文件读取。
立即学习“Java免费学习笔记(深入)”;
序列化示例:
Person person = new Person("张三", 25);try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("person.ser"))) {
oos.writeObject(person);
} catch (IOException e) {
e.printStackTrace();
}
反序列化示例:
Person deserializedPerson = (Person) ois.readObject();
System.out.println(deserializedPerson.getName() + ", " + deserializedPerson.getAge());
} catch (IOException | ClassNotFoundException e) {
e.printStackTrace();
}
注意:反序列化时可能抛出 ClassNotFoundException,尤其在类路径找不到对应类时。
序列化过程中需要注意的关键问题
实际开发中,以下几点容易被忽略但非常重要:
-
serialVersionUID 的作用:JVM 使用该字段验证序列化前后类的一致性。如果不显式定义,系统会根据类名、字段等生成,一旦类结构变化可能导致 InvalidClassException。建议手动指定:
private static final long serialVersionUID = 1L; -
transient 关键字:用于标记不希望被序列化的字段。例如密码、临时缓存等敏感或非持久化数据。
private transient String password; - 父类的序列化处理:如果父类实现了 Serializable,子类自动可序列化。若父类未实现,则父类字段不会被序列化,即使子类实现了接口。
-
单例模式与序列化安全:默认反序列化会创建新实例,破坏单例。可通过添加 readResolve() 方法防止:
private Object readResolve() {
return INSTANCE; // 返回原有实例
}
常见应用场景与注意事项
对象序列化常用于以下场景:
- 远程方法调用(RMI)中传递对象参数
- 将对象状态保存到本地文件或数据库
- Web应用中保存Session对象
- 分布式系统中跨节点传输对象
但也要注意:
- 序列化后的文件不加密,敏感信息需额外处理
- 性能开销较大,不适合高频操作
- 不同JVM版本或厂商可能影响兼容性
基本上就这些。掌握好 Serializable 的使用规则和陷阱,能有效避免运行时异常和数据不一致问题。不复杂但容易忽略细节。










