Java对象为什么要实现Serializable接口?

Serializable接口的使用

一.引入问题

一般情况下,我们在定义实体类时会继承Serializable接口,类似这样:

import java.io.Serializable;

public class Person implements Serializable {
    String name;
    int age;

    public Person(String name,int age){
        this.name=name;
        this.age=age;
    }

    @Override
    public String toString() {
        return String.format("Person{ %s : %d }",name,age);
    }
}

通过以上我们可以发现,我们在定义一个实体类的时候实现了这个Serializable接口,那么实现这个接口到底有什么用呢?细心的你还会发现,我们还定义了个serialVersionUID变量,那么这个变量又有什么作用呢?

二.什么是Serializable接口

它是一个对象序列化的接口,一个类只有实现了Serializable接口,它的对象才可以被序列化。不实现此接口的类的任何字段(属性)都不能序列化和反序列化。

三.什么是序列化

序列化是将对象状态转换为可保持或传输的格式的过程。与序列化相对的是反序列化,它将流又转换为对象。这两个过程结合起来,可以轻松的存储和传输数据


1.序列化(场景):

以上面提到的Person类为例。这个类中的两个字段name和age在程序运行后都在堆内存中,程序执行完毕后内存得到释放,name和age的值也不复存在。如果现在计算机要把这个类的实例发送到另一台机器、或是想保存这个类的实例到数据库(持久化对象为目的),以便以后再取出来用。这时就需要对这个类进行序列化,便于传送或保存。用的时候再反序列化重新生成这个对象的实例即可。

2.反序列化(场景):

以搬桌子为例,桌子太大了不能通过比较小的门,我们要把它拆了再运进去,这个拆桌子的过程就是序列化。同理,反序列化就是等我们需要用桌子的时候再把它组合起来,这个过程就是反序列化。


序列化前的对象和反序列化后得到的对象,内容是一样的(且对象中包含的引用也相同),但两个对象的地址不同,就相当于是两个对象。换句话说,序列化操作可以实现对任何可Serializable对象的”深度复制(deep copy)"。

四.为什么要序列化对象

** 把对象转换为字节序列的过程称为对象的序列化
** 把字节序列恢复为对象的过程称为对象的反序列化

五.什么情况下需要序列化

当我们需要把对象的状态信息通过网络进行传输,或者需要将对象的状态信息持久化,以便将来使用。这些情况都需要把对象进行序列化。而实现Serializable接口,可以存储对象在存储介质中,以便在下次使用的时候可以很快捷的重建一个副本。

六.Serializable的源码

public interface Serializable{
}

序列化接口没有方法或字段,仅用于标识可序列化的语义。 (一个接口里面什么内容都没有,我们可以将它理解成一个标识接口)

---------怎么理解标识的含义?
答:比如在课堂上有位学生遇到一个问题,于是举手向老师请教,这时老师帮他解答,那么这位学生的举手其实就是一个标识。自己解决不了问题,请教老师帮忙解决。在Java中的这个Serializable接口其实是给JVM看的,通知JVM,我不对这个类做序列化了,你JVM帮我序列化就好了。

----------什么是JVM呢?
答:JVM是Java Virtual Machine(Java虚拟机)的缩写,JVM是一种用于计算设备的规范,它是一个虚构出来的计算机,是通过在实际的计算机上仿真模拟各种计算机功能来实现的。

七.为什么要定义serialversionUID变量

在反序列化的过程中则需要使用serialVersionUID来确定由那个类来加载这个对象,所以我们在实现Serializable接口的时候,一般还要去显示地定义serialVersionUID,例如:

private static final long serialVersionUID = 1L;

serialVersionUID是用来辅助对象的序列化与反序列化的。原则上序列化 后的数据当中的serialVersionUID与当前类当中的serialVersionUID是一致的时候,那么该对象才能被反序列化成功。如果我们没有自己声明一个serialVersionUID变量,接口会默认生成一个serialVersionUID。但是强烈建议用户自定义一个serialVersionUID,因为默认的serialVersinUID对于class的细节非常敏感,若没有自定义这个变量,反序列化时可能会导致InvalidClassException这个异常。


serialVersionUID的详细的工作机制:在序列化的时候系统将serialVersionUID写入到序列化的文件中去,当反序列化的时候系统会先去检测文件中的serialVersionUID是否跟当前的文件的serialVersionUID是否一致,如果一致则反序列化成功,否则就说明当前类跟序列化后的类发生了变化。比如是成员变量的数量或者是类型发生了变化,那么在反序列化时就会发生crash,并且会报出错误。


总结:如果我们在序列化中没有显示地声明serialVersionUID,则序列化运行时将会根据该类的各个方面计算该类默认的serialVersionUID值。但是,Java官方强烈建议所有要序列化的类都最好显示地声明serialVersionUID字段,因为如果高度依赖于JVM默认生成的serialVersionUID,可能会导致其与编译器的实现细节耦合,这样可能会导致在反序列化的过程中发生意外的InvalidClassException异常。因此,为了保证跨不同Java编译器实现的serialVersionUID值的一致,实现Serializable接口的必须显示地声明serialVersionUID字段。

此外serialVersionUID字段地声明要尽可能使用private关键字修饰,这是因为该字段的声明只适用于声明的类,该字段作为成员变量被子类继承是没有用的。 有个特殊的地方需要注意的是,数组类是不能显示地声明serialVersionUID的,因为它们始终具有默认计算的值,不过数组类反序列化过程中也是放弃了匹配serialVersionUID值的要求。

八.对Person类对象进行序列化和反序列化

          **我们把Person对象写到一个文件中然后读出来**

1.序列化

import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectOutputStream;

public class UsePerson {
    public static void main(String[] args) throws IOException {
        Person p=new Person("张三",18);
        ObjectOutputStream oos=new ObjectOutputStream(new FileOutputStream("Person对象.obj"));
        oos.writeObject(p);
        System.out.println("序列化成功");
        oos.close();
    }
}

2.反序列化

import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.ObjectInputStream;

public class UsePerson1 {
    public static void main(String[] args) throws IOException, ClassNotFoundException {
        ObjectInputStream ois=new ObjectInputStream(new FileInputStream("Person对象.obj"));
        Person p=(Person)ois.readObject();
        System.out.println(p);
        System.out.println("反序列化成功");
        ois.close();
    }
}

通过序列化将Person对象写到一个文件中,然后通过反序列化将Person对象从文件中进行恢复,恢复后得到的内容和之前完全一样,但是两者是不同的对象。

九.注意事项

a)序列化时,只对对象的状态进行保存,而不管对象的方法;
b)当一个父类实现了序列化时,该子类自动实现序列化,不需要显式实现Serializable接口;
c)并非所有的对象都可以序列化。
d) 序列化会忽略静态变量,即序列化不保存静态变量的状态。静态成员属于类级别的,不能被序列化。同时添加了static、transient关键字后的变量也不能被序列化。

十.总结

对于JVM来说,要进行持久化的类必须要有一个标记,只有持有这个标记JVM才允许类创建的对象可以通过其IO系统转换为字节数据,从而实现持久化,而这个标记就是Serializable。

  • 3
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值