Serializable接口的作用

一.什么是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值的要求。

当序列化/反序列化多个对象时:

import java.io.FileOutputStream;
import java.io.ObjectOutputStream;
import java.util.ArrayList;
import java.util.List;

/*
    一次序列化多个对象:将多个对象放到集合当中,然后序列化集合
    参与序列化的ArrayList集合以及集合中的元素User都需要实现java.io.Serializable接口
 */

public class ObjectOutputStreamTest02 {
    public static void main(String[] args) throws Exception{
        List<User> userList = new ArrayList<>();
        userList.add(new User(888,"张三",18));
        userList.add(new User(222,"李四",19));
        userList.add(new User(333,"王五",20));
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("users"));

        //序列化一个集合,这个集合对象中放了很多其他对象
        oos.writeObject(userList);
        System.out.println("序列化成功");
        oos.flush();
        oos.close();
    }
}


import java.io.FileInputStream;
import java.io.ObjectInputStream;
import java.util.List;
/*
    反序列化集合
 */
public class ObjectInputStreamTest02 {
    public static void main(String[] args) throws Exception {
        ObjectInputStream ois = new ObjectInputStream(new FileInputStream("Users"));
      /*  Object  object = ois.readObject();
        System.out.println(object instanceof List);*///true

        List<User> list = (List<User>) ois.readObject();
        System.out.println("反序列化成功");
        for (User user : list) {
            System.out.println(user);
        }

        ois.close();
    }
}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值