java序列化接口,为什么提倡所有类都实现?

        在java基础类中,很多类都实现了java.io.Serializable接口,诸如常见的Integer、String、List,这个接口用处是什么,为什么在公司中都规定所有类都实现Serializable接口呢?

什么是序列化和反序列化?

Java提供了一种对象序列化的机制

        序列化:把java对象转换为字节序列的过程;在网络传输对象或者将对象持久化到文件中时,将对象转换成有序字节流,字节流中包含对象完整的状态数据信息,保证对象的完整性和可传递性。

        反序列化:把字节序列转换为java对象的过程;程序在文件或者网络传输中,获取到字节流后,根据字节流中包含的对象状态数据信息,重建java对象的过程。

可以将序列化想象成人去银行存钱的过程

将现金放入ATM机中,存入银行卡中的过程可当作序列化的过程,

现金 => ATM => 银行卡   
java对象  => JVM => 字节序列

将银行卡中的钱通过ATM取出的过程,可以当作反序列化的过程

银行卡 => ATM => 现金 
字节序列  => JVM => java对象

为什么要序列化?

        在文件存储和网络传输时,所有的文件,视频也好,文本,图片,音频也好,都是以二进制序列的形式进行存储或传输。

        那么在java两个进程中进行通信的时候,就需要我们也同样利用二进制进行传输,而往往客户端在接收到服务端传输过来的字节序列后,由jvm将字节序列转换为java对象,已供程序使用。

        由于序列化是jvm提供的一种机制,所有序列化可以保证在一个平台中序列化后的对象,可以在另一个平台上进行重建出来


Java如何实现序列化和反序列化?

JDK序列化接口


java.io.ObjectOutputStream:表示对象输出流;

try (ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream("user.txt"))){
    objectOutputStream.writeObject(userOut);
}

初始化一个指向user.text文件的输出流


objectOutputStream.writeObject(userOut);方法可以将对象二进制输出到指向的user.text中

java.io.ObjectInputStream:表示对象输入流;

try(ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream("user.txt"))) {
   User user = (User) objectInputStream.readObject();
}

初始化一个指向uset.text文件的输入流
        objectInputStream.readObject();方法可以将文件user.text中的字节序列读取并重建对象,重建后可直接强制转换为序列化前的对象

实例化对象的要求

        只有自身类或者父类、实现类实现了Serializable接口的类对象才能被序列化,否则抛出
java.io.NotSerializableException异常,需要注意的是,如果类的成员变量属于类对象,那么也需要实现Serializable接口才能实例化,否则也将抛出异常。

        若实现了Serializable接口,在序列化时,将递归将对象的属性(方法不会序列化)转化为字节序列:

        ObjectOutputStream采用默认的序列化方式,对User对象的非transient的实例变量进行序列化。

        ObjcetInputStream采用默认的反序列化方式,对User对象的非transient的实例变量进行反序列化。

序列化和反序列化代码实例:

import java.io.*;

public class TestSerializable {
    public static void main(String[] args) throws IOException, ClassNotFoundException {
        User userOut = new User("榴莲java", 23,"a");
        try (ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream("user.txt"))){
            objectOutputStream.writeObject(userOut);
        }

        try(ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream("user.txt"))) {
            User userPut = (User) objectInputStream.readObject();
            System.out.println(userPut.toString());
        }

    }

    static class User implements Serializable {
        private static final long serialVersionUid = 9992L;
        private transient String name;
        private Integer age;
        public User(String name, Integer age, String sex) {
            this.name = name;
            this.age = age;
        }
        public String toString(){
            return "name:" + this.name + "\tage:" + this.age;
        }
    }
}

执行结果:

name:null age:23

transient关键字详解

        大家也看到了,上述代码执行结果中name变量并未实例化,原因是name被transient关键字修饰,用transient关键字标记的成员变量将不参与类的实例化和反实例化过程。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

JAVA新视界

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值