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

2023年 9月 12日 51.6k 0

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

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关键字标记的成员变量将不参与类的实例化和反实例化过程。

相关文章

JavaScript2024新功能:Object.groupBy、正则表达式v标志
PHP trim 函数对多字节字符的使用和限制
新函数 json_validate() 、randomizer 类扩展…20 个PHP 8.3 新特性全面解析
使用HTMX为WordPress增效:如何在不使用复杂框架的情况下增强平台功能
为React 19做准备:WordPress 6.6用户指南
如何删除WordPress中的所有评论

发布评论