父类实现Serializable时,子类可序列化其字段;若父类未实现,需自定义writeObject/readObject手动处理父类字段以避免数据丢失。

Java对象序列化时,父类字段的处理方式取决于父类是否实现了Serializable接口。序列化机制对父类和子类字段的处理有明确规则,理解这些规则有助于避免序列化过程中出现意外的数据丢失或异常。
父类实现Serializable接口
如果父类也实现了Serializable接口,那么整个继承链上的字段都可以被正常序列化。
- 子类和父类中所有非静态、非瞬态(
transient)的字段都会被自动序列化。 - 序列化过程会递归处理所有可序列化的字段,包括从父类继承的字段。
- 反序列化时,不需要调用构造函数,字段值直接从流中恢复。
class Person implements Serializable {
private String name;
// getter/setter
}
class Student extends Person implements Serializable {
private int studentId;
// getter/setter
}
此时name和
studentId都能被正常序列化。接口,即使子类实现了,父类的字段也不会被序列化。父类未实现Serializable接口
如果父类没有实现
Serializable
立即学习“Java免费学习笔记(深入)”;
- 序列化时仅保存子类自身定义的可序列化字段。
- 反序列化创建对象时,会调用父类无参构造函数初始化继承的字段。
- 若父类没有无参构造函数,会抛出
InvalidClassException。
这种情况下,父类字段在反序列化后会回到“初始状态”,而不是序列化前的值,可能导致数据丢失。
控制父类字段序列化行为
当父类不可序列化但需要保留其字段状态时,可通过以下方式干预:
- 在子类中显式添加
writeObject和readObject方法。 - 在
writeObject中手动将父类字段写入输出流。 - 在
readObject中从输入流读取并恢复父类字段。
private void writeObject(ObjectOutputStream out) throws IOException {
out.defaultWriteObject(); // 先序列化子类字段
out.writeObject(super.getName()); // 手动写父类字段
}
private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException {
in.defaultReadObject(); // 恢复子类字段
setParentField((String) in.readObject()); // 手动恢复父类字段
}
基本上就这些。父类是否可序列化直接影响字段能否被保存,尤其在继承结构中需特别注意构造函数和字段初始化逻辑。不复杂但容易忽略。










