Serializable
Serializable接口位于java.io.Serializable包中,一般在创建实体类的时候会去实现这个接口,目的是为了序列化。
什么是序列化和反序列化
序列化指的是把对象转化成字节序列的过程,就是把内存中的对象转换成一连串的字节bytes描述的过程。
反序列化就是将字节bytes数据恢复为对象的过程。
常见场景
-
需要将内存中对象状态数据保存到文件或数据库中时,需要进行序列化。这个比较直观的就是通过ObjectOutputStream和FileOutputStream向文件中保存对象时,实现了Serializable接口的对象会以二进制数据的形式保存到文件中,而为实现Serializable接口的对象则会直接报错,无法保存。
下图实现了Serializable的对象保存成功(虽然乱码)
下图为实现Serializable接口的对象保存失败
-
网络通信时需要用套接字在网络中传送对象时,如我们使用RPC协议进行网络通信时。还没用到过,先不描述。
serialVersionUID
进行持久化时,只有带有一个特殊标记的类创建的对象才会被jvm通过IO系统转换为字节数据,标记便是Serializable接口。反序列化的过程需要使用serialVersionUID来确定该由哪个类来加载将要被反序列化的对象,所以实现Serializable接口的时候需要一般需要显式指明serialVersionUID,比如这样:
private static final long serialVersionUID = 1L;
如果在进行反序列化前,将某个实体类原有的serialVersionUID改变,在进行反序列化时就会报InvalidClassException异常。
如果没有指定serialVersionUID,在进行序列化时,JVM会通过计算给该类一个默认的serialVersionUID值。
但是不建议不指定serialVersionUID,高度依赖于JVM默认生成serialVersionUID,可能会导致其与编译器的实现细节耦合,这样可能会导致在反序列化的过程中发生意外的InvalidClassException异常。个人认为应该是指不同编译器编译的不同类可能产生一个相同的默认serialVersionUID。
另外尽量用private来修饰serialVersionUID。