序列化有两个作用:数据持久化和远程通信传输
序列化是如何实现的?
一般实体类实现Serializable(常用)或者Externalizable(少用)即可
依靠流stream->inputStream、outputStreadm->ObjectInputStream、ObjectInputStream
使用步骤
- 写 序列化对象
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("C:\\SerializableObject.out"));
oos.writeObject(new User());
- 读 反序列化对象
ObjectInputStream ois= new ObjectInputStream(new FileInputStream("C:\\SerializableObject.out"));
User user = (User) ois.readObject();
如果用户重写了readObject和writeObject方法,就用自己的,不然就用默认原生的逻辑。
序列化版本号serialVersionUID
经常在看源码的时候会看到
private static final long serialVersionUID = -xxxxL; 之类的代码*(顺便说一句,static的变量和transient的一样不会被序列化)*
这是什么作用呢?
serialVersionUID的计算方式是根据类名,接口名,方法和属性等来生成,只要类有变化,添加一个空格或者一个字段都会有变化,Java会重新计算UID。那么已经序列化的对象,在类文件改动后,反序列化的时候,会出于保护机制而报错。serialVersionUID 如果不指定,java会执行默认摘要算法,生成serialVersionUID,每次重新编译都有可能不同,从而导致反序列化的报错。
因此,为了减少bug概率,需要序列化的类要指定显式serialVersionUID的值,这样改动类文件后,不同版本的同个类,也可以正常反序列化。这是一个规范也是一个习惯。