1.首先你要先搞明白什么是序列化和反序列化
序列化:将java对象转化为字节序列的过程叫做序列化
反序列化:将字节序列转化为java对象的过程叫做反序列化
java对象序列化主要有两大用途:
第一种:将java对象序列化成字节序列保存到磁盘上,需要的时候反序列化成java对象
第二种:将java对象序列化成字节序列在网络上传输,网络传输的都是字节序列。
2.serialVersionUID干什么用的
java对象序列化必须有一个serialVersionUID,如果自己没有声明serialVersionUID,那么接口会默认生成一个serialversionUID
反序列化的时候会用这个serialVersionUID,对比字符序列的这个serialVersionUID和类中的serialVersionUID是否一样,
如果不一样就会报异常 java.io.InvalidClassException
Exception in thread "main" java.io.InvalidClassException: com.fengqing.test32.Person; local class incompatible: stream classdesc serialVersionUID = 8350891266866170016, local class serialVersionUID = 8350891266866170017
at java.io.ObjectStreamClass.initNonProxy(ObjectStreamClass.java:562)
at java.io.ObjectInputStream.readNonProxyDesc(ObjectInputStream.java:1583)
at java.io.ObjectInputStream.readClassDesc(ObjectInputStream.java:1496)
at java.io.ObjectInputStream.readOrdinaryObject(ObjectInputStream.java:1732)
at java.io.ObjectInputStream.readObject0(ObjectInputStream.java:1329)
at java.io.ObjectInputStream.readObject(ObjectInputStream.java:351)
at com.fengqing.test32.TestPerson.main(TestPerson.java:19)
强烈建议用户自定义一个serialVersionUID,因为默认的serialVersinUID对于类的细节非常敏感
3.用户自定义serialVersionUID的好处
如果我们在反序列化之前把类中的内容做了修改,不如多加了一个属性,反序列化时就会报异常,因为类中的内容变了之后
默认生成的serialVersionUID和之前的不一样了。
如果是用户自定义的serialVersionUID,即使你改变了类中的内容,只有serialVersionUID不变,发序列化一样能成功。
举例
public class Person implements Serializable{
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
public class TestPerson {
public static void main(String[] args) throws Exception {
Person p = new Person();
p.setName("fengqing");
//序列化
/*FileOutputStream fos = new FileOutputStream("person.txt");
ObjectOutputStream oos = new ObjectOutputStream(fos);
oos.writeObject(p);*/
//反序列化
FileInputStream fis = new FileInputStream("person.txt");
ObjectInputStream ois = new ObjectInputStream(fis);
Person person = (Person)ois.readObject();
System.out.println(person.getName());
}
}
反序列化之前加一个address属性
public class Person implements Serializable{
private String address;
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
发序列化就会失败,
如果是自定义serialVersionUID就不会失败
public class Person implements Serializable{
private static final long serialVersionUID = 8350891266866170016L;
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}