关于Serializable的一篇笔记

最近在学习代码规范的时候POJO中用到了VO, DTO, entity这些东西, 然后发现实体类的代码中都会有 implements serializable 这段代码, 然后来网上学学这段代码的用处, 发现很多人都是和我一样接触到代码规范来进行的记录
在这里插入图片描述
在程序中为了能直接以 Java 对象的形式进行保存(持久化保存, 序列化就是保存, 反序列化就是读取),然后再重新得到该 Java 对象,这就需要序列化能力。序列化其实可以看成是一种机制,按照一定的格式将 Java 对象的某状态转成介质可接受的形式,以方便存储或传输。其实想想就大致清楚基本流程,序列化时将 Java 对象相关的类信息、属性及属性值等等保存起来,反序列化时再根据这些信息构建出 Java 对象。而过程可能涉及到其他对象的引用,所以这里引用的对象的相关信息也要参与序列化。

注: Serializable主要作用将类的实例持久化保存,序列化就是保存,反序列化就是读取。保存也不一定保存在本地,也可以保存到远方。类一定要实现Serializable才可以(也可以是Externalizable 接口)

序列化的作用?

  • 提供一种简单又可扩展的对象保存恢复机制。
  • 对于远程调用,能方便对对象进行编码和解码,就像实现对象直接传输。
  • 可以将对象持久化到介质中,就像实现对象直接存储。
  • 允许对象自定义外部存储的格式。

什么时候要进行序列化?
在存储时需要序列化,这是肯定的。大家知道的是序列化是将对象进行流化存储,我们有时候感觉自己在项目中并没有进行序列化操作,也一样是存进去了,那么对象需要经过序列化才能存储的说法,似乎从这儿就给阉割了。事实究竟是怎样的呢?

首先看我们常用的数据类型类声明:
public final class String implements java.io.Serializable, Comparable<String>, CharSequence
public class Date implements java.io.Serializable, Cloneable, Comparable

而像其他int、long、boolean类型等,都是基本数据类型,数据库里面有与之对应的数据结构。从上面的类声明来看,我们以为的没有进行序列化,其实是在声明的各个不同变量的时候,由具体的数据类型帮助我们实现了序列化操作。

到这儿的时候,就又有一个问题,既然实体类的变量都已经帮助我们实现了序列化,为什么我们仍然要显示的让类实现serializable接口呢?

注意以上的说法:首先,序列化的目的有两个,

  1. 第一个是便于存储,
  2. 第二个是便于传输。

我们一般的实体类不需要程序员再次实现序列化的时候,请想两个问题:第一:存储媒体里面,是否是有其相对应的数据结构?第二:这个实体类,是否需要远程传输(或者两个不同系统甚至是分布式模块之间的调用)?

如果有人打开过Serializable接口的源码,就会发现,这个接口其实是个空接口,那么这个序列化操作,到底是由谁去实现了呢?其实,看一下接口的注释说明就知道,当我们让实体类实现Serializable接口时,其实是在告诉JVM此类可被序列化,可被默认的序列化机制序列化。

public interface Serializable {
}

然后来写代码实践一下看看Serializable是如何序列化Java对象的.创建一个类SClass(序列化类),增加name和age两个属性,并创建Getter和Setter方法。

public class SClass {
    private String name;
    private Integer age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }
}

再创建一个测试类,通过ObjectOutputStream将一个SClass对象写入文件中,这个实际上就是一种序列化的过程;再通过ObjectInputSream将SClass对象读取出来,这个实际上就是一个反序列化的过程。

public class Test {
    public static void main(String[] args) {
        // 初始化
        SClass sclass = new SClass();
        sclass.setName("王二");
        sclass.setAge(18);
        System.out.println(sclass);
        // 把对象写到文件中
        try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("file"));){
            oos.writeObject(sclass);
        } catch (IOException e) {
            e.printStackTrace();
        }
        // 从文件中读出对象
        try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream(new File("file")));){
            SClass sclass1 = (SClass) ois.readObject();
            System.out.println(sclass1);
        } catch (IOException | ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

运行一下会发现出现报错, 这是由于SClass没有实现Serializable接口,所以系统会报错。
在这里插入图片描述
顺着堆栈信息,我们来看一下 ObjectOutputStream 的 writeObject0() 方法。其部分源码如下:

if (obj instanceof String) {
    writeString((String) obj, unshared);
} else if (cl.isArray()) {
    writeArray(obj, desc, unshared);
} else if (obj instanceof Enum) {
    writeEnum((Enum<?>) obj, desc, unshared);
} else if (obj instanceof Serializable) {
    writeOrdinaryObject(obj, desc, unshared);
} else {
    if (extendedDebugInfo) {
        throw new NotSerializableException(
            cl.getName() + "\n" + debugInfoStack.toString());
    } else {
        throw new NotSerializableException(cl.getName());
    }
}

这段代码的意思是,ObjectOutPutStream 在序列化的时候,会判断对象的类型,如果不是字符串、数组、枚举、Serializable的湖锕,会抛出NotSerializableException
但是,如果SClass实现了Serializable接口的话,就可以被序列化和反序列化了。

具体是怎么序列化的呢?
ObjectOutputStream为例,它在序列化的时候会依次调用

writeObject()→writeObject0()→writeOrdinaryObject()→writeSerialData()→invokeWriteObject()→defaultWriteFields()

defaultWriteFields方法为真正将对象序列化的接口。



那怎么反序列化呢?
ObjectInputStream 为例,它在反序列化的时候会依次调用

readObject()→readObject0()→readOrdinaryObject()→readSerialData()→defaultReadFields()

defaultReadFields方法为真正将对象反序列化的接口

所以Serializable接口仅仅是起到了标识的作用,告诉程序,他可以被序列化。

补充:
1.static 和 transient 修饰的字段是不会被序列化的。
因为序列化保存的是对象的状态,而 static 修饰的字段属于类的状态,因此可以证明序列化并不保存 static 修饰的字段。
transient 的中文字义为“临时的”(论英语的重要性),它可以阻止字段被序列化到文件中,在被反序列化后,transient 字段的值被设为初始值,比如 int 型的初始值为 0,对象型的初始值为 null。

2.除了 Serializable 之外,Java 还提供了一个序列化接口 Externalizable(念起来有点拗口)。

3.serialVersionUID 被称为序列化 ID,它是决定 Java 对象能否反序列化成功的重要因子。在反序列化时,Java 虚拟机会把字节流中的 serialVersionUID 与被序列化类中的 serialVersionUID 进行比较,如果相同则可以进行反序列化,否则就会抛出序列化版本不一致的异常。

我的其他文章:
关于serialVersionUID的作用解释说明

参看文章(侵删):
https://www.cnblogs.com/tentacion/p/11356174.html
Java中Serializable接口的作用是什么
谈谈实现Serializable接口的作用和必要性

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值