Java的序列化与反序列化

序列化是干啥用的?

  • 序列化:把Java对象转换为字节序列。
  • 反序列化::把字节序列恢复为原先的Java对象。
@Data
@Builder
public class Student implements Serializable {
    private String name;
    private Integer age;
    private Integer score;

}

package com.lsh.util;

import com.lsh.model.Student;

import java.io.*;

/**
 * @author :LiuShihao
 * @date :Created in 2020/12/15 5:55 下午
 * @desc :
 */
public class SerializableUtil {
    public static void main(String[] args) throws Exception {
        serializ();
        deserializ();
    }
    
    public static void serializ() throws IOException {
        // 创建Student对象  这里使用了 lombok 中的建造者模式构建对象
        Student student = Student.builder().name("LiuShihao").age(18).score(100).build();
        ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream(new File("student.txt")));
        objectOutputStream.writeObject(student);
        objectOutputStream.close();
        System.out.println("序列化成功! 已生成student.txt文件");
    }

    public static void deserializ() throws IOException, ClassNotFoundException {
        ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream(new File("student.txt")));
        Student student = (Student) objectInputStream.readObject();
        System.out.println("==================================");
        System.out.println("反序列化结果:");
        System.out.println(student);
    }

}

在这里插入图片描述

注意:
序列化的txt文件打开是为乱码:
在这里插入图片描述

因为乱码才是正常的,序列化和反序列话是基于二进制流的,这个二进制流不受制于任何字符编码格式
序列化是把Student对象的信息以二进制存储在文件obj.bat中,不是以特定的字符编码格式输出的
用文本编辑器打开自然是乱码。只有通过反序列话才能将存储的二进制读取出来,然后显示在控制台上。

实现Serializable接口

在这里插入图片描述
点进这个接口发现这是一个空接口。

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

serialVersionUID号有什么用?

在这里插入图片描述

在这里插入图片描述

第一个问题:

序列化ID,可以看成是序列化和反序列化过程中的“暗号”,在反序列化时,JVM会把字节流中的序列号ID和被序列化类中的序列号ID做比对,只有两者一致,才能重新反序列化,否则就会报异常来终止反序列化的过程。

第二个问题:

如果在定义一个可序列化的类时,没有人为显式地给它定义一个serialVersionUID的话,则Java运行时环境会根据该类的各方面信息自动地为它生成一个默认的serialVersionUID,一旦更改了类的结构或者信息,则类的serialVersionUID也会跟着变化!
所以,为了serialVersionUID的确定性,写代码时还是建议,凡是implements Serializable的类,都最好人为显式地为它声明一个serialVersionUID明确值!
当然,如果不想手动赋值,你也可以借助IDE的自动添加功能。

使用IntelliJ IDEA自动生成serialVersionUID:

在这里插入图片描述
在这里插入图片描述

两种特殊情况

在这里插入图片描述
对于第一点,因为序列化保存的是对象的状态而非类的状态,所以会忽略static静态域也是理所应当的。
对于第二点,就需要了解一下transient修饰符的作用了。如果在序列化某个类的对象时,就是不希望某个字段被序列化(比如这个字段存放的是隐私值,如:密码等),那这时就可以用transient修饰符来修饰该字段。
比如在之前定义的Student类中,加入一个密码字段,但是不希望序列化到txt文本,则可以:
在这里插入图片描述

序列化的受控和加强

束性加持

从上面的过程可以看出,序列化和反序列化的过程其实是有漏洞的,因为从序列化到反序列化是有中间过程的,如果被别人拿到了中间字节流,然后加以伪造或者篡改,那反序列化出来的对象就会有一定风险了,毕竟反序列化也相当于一种“隐式的”对象构造,因此我们希望在反序列化时,进行受控的对象反序列化动作。那怎么个受控法呢?
笞案就是:自行编写 readobject()函数,用于对象的反序列化构造,从而提供约束性。既然自行编写 readobject()函数,那就可以做很多可控的事情:比如各种判断工作。还以上面的 Student:类为例,一般来说学生的成绩应该在0-100之间,我们为了防止学生的考试成绩在反序列化时被别人簒改成一个奇葩值,我们可以自行编写 readobject()函数用于反序列化的控制

单例模式增强

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Liu_Shihao

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

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

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

打赏作者

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

抵扣说明:

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

余额充值