凡是实现Serializable接口的类都有一个表示序列化版本标识符的静态常量:
public class XXX implements Serializable {
private static final long serialVersionUID = 3981882461445732799L;
...
}
serialVersionUID用来表明类的不同版本间的兼容性,其目的是以序列化对象进行版本控制,有关各版本反序加化时是否兼容。
如果类没有显示定义这个静态变量,它的值是JRE根据类的内部细节自动生成的。若类做了修改,serialVersionUID 可能发生变化。故建议,显式声明。
举一个例子
比如我们有一个Person类,实现了Serializable接口:
public class Person implements Serializable {
String name;
int age;
}
我们将它的一个实例O进行序列化并保存在一个文件A里。
随后,因为一些原因,我们对Person类进行了一些修改:删除了age,并重写了toString方法。
public class Person implements Serializable {
String name;
@Override
public String toString() {
return "Person{" +
"name='" + name + ',' +
'}';
}
}
然后,我们从文件A中将实例O反序列化,将其转换回Person类,此时程序会抛出InvalidClassException异常。
这是因为Java会根据serialVersionUID的值是否一致来判断这个Person是否是当初的那个Person,
Person没有显示定义serialVersionUID的值,它的值是JRE根据类的内部细节自动生成,
当我们对Person进行修改后,Person的serialVersionUID也会发生变化,
所以当先前被序列化的实例回来后,出现了serialVersionUID不匹配的问题,JRE会认为新的Person无法兼容旧的实例O。
所以我们可以在被序列化的对象中显式地声明serialVersionUID的值,使之固定不变,从而使得由旧对象创建的实例可以被新的对象兼容。