Java序列化接口,为什么提倡所有类都实现?
创始人
2025-07-01 08:01:15
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关键字标记的成员变量将不参与类的实例化和反实例化过程。

相关内容

热门资讯

如何允许远程连接到MySQL数... [[277004]]【51CTO.com快译】默认情况下,MySQL服务器仅侦听来自localhos...
如何利用交换机和端口设置来管理... 在网络管理中,总是有些人让管理员头疼。下面我们就将介绍一下一个网管员利用交换机以及端口设置等来进行D...
施耐德电气数据中心整体解决方案... 近日,全球能效管理专家施耐德电气正式启动大型体验活动“能效中国行——2012卡车巡展”,作为该活动的...
20个非常棒的扁平设计免费资源 Apple设备的平面图标PSD免费平板UI 平板UI套件24平图标Freen平板UI套件PSD径向平...
德国电信门户网站可实时显示全球... 德国电信周三推出一个门户网站,直观地实时提供其安装在全球各地的传感器网络检测到的网络攻击状况。该网站...
为啥国人偏爱 Mybatis,... 关于 SQL 和 ORM 的争论,永远都不会终止,我也一直在思考这个问题。昨天又跟群里的小伙伴进行...
《非诚勿扰》红人闫凤娇被曝厕所... 【51CTO.com 综合消息360安全专家提醒说,“闫凤娇”、“非诚勿扰”已经被黑客盯上成为了“木...
2012年第四季度互联网状况报... [[71653]]  北京时间4月25日消息,据国外媒体报道,全球知名的云平台公司Akamai Te...