java之序列化与反序列化的详细解析(全)

前言

经常看到一些源码或者代码项目中继承一个Serializable 接口,但是究其原因有时候却很难讲清楚

这篇文章涉及一些IO流可看我之前的文章
java NIO从入门到精通(全)

1. 定义

序列化(Serialize):内存当中的java对象放到硬盘文件中,java对象存储到文件中,将java对象的状态保存下来的过程,需要使用ObjectOutputStream类

反序列化(DeSerialize):将硬盘上的数据重新恢复到内存中,恢复成java对象。需要使用ObjectInputStream类

通俗的说,序列化就是通过网络中的管道,切成一个一个小的数据包(有编号,拆分)放到硬盘文件。一个个小的数据包(组装)恢复到内存中,就是反序列化

或者也可以这样理解

序列化是指把一个Java对象变成二进制内容,因为要在网络中传输,而且要快便捷,本质上是一个byte[]数组,序列化后可以将其数组保存到网络中传输
。反序列化,即把一个二进制内容(也就是byte[]数组)变回Java对象

对象的序列化主要有两种用途:

  • 把对象的字节序列永久地保存到硬盘上,通常存放在一个文件中;(持久化对象)
  • 在网络上传送对象的字节序列。(网络传输对象)

2. 细节

java 的transient关键字为我们提供了便利,你只需要实现Serilizable接口,将不需要序列化的属性前添加关键字transient,序列化对象的时候,这个属性就不会序列化到指定的目的地中

2.1 API接口

ObjectOutputStream和ObjectInputStream类

序列化和反序列化需要使用的两个类的api接口:

①java.io.ObjectInputStream:对象输入流
使用该类的readObject()方法从输入流中读取字节序列
将字节序列反序列化为一个对象

②java.io.ObjectOutputStream:对象输出流
使用该类的writeObject(Object obj)方法将传入的obj对象进行序列化
得到的字节序列写入输出流中输出

Serializable接口

  • 一般出现这个错误java.io.NotSerializableException,说明创建的实体类对象对象没有继承序列化
  • 参与序列化和反序列化的实体类对象,必须实现Serializable接口
  • 参与序列化的ArrayList集合以及集合中的元素User都需要实现 java.io.Serializable接口。

通过查看其Serializable接口的源码,只是一个标志接口:

public interface Serializable {
}

接口中的代码没有实体的,但却要继承,主要是起到一个标识的作用,jvm看到实体类继承了Serializable这个接口,就会自动为该类生成一个版本序列号

2.2 版本序列号

所谓的版本序列号

java虚拟机识别一个类的时候先通过类名,如果类名一致,再通过序列化版本号

同一个文件中源代码改动之后,需要重新编译,编译之后生成了全新的字节码文件,并且class文件再次运行的时候,java虚拟机生成的序列化版本号也会发生相应的改变。所以一般写上序列号之后,即使改动了之后,也可以识别到该类,一般要写上序列号

一般这个序列号写在继承 Serializable接口的的实体类下面

private static final long serialVersionUID = 1L; // java虚拟机识别一个类的时候先通过类名,如果类名一致,再通过序列化版本号。

2.3 idea自动生成版本序列号

建议将序列化版本号手动的写出来。不建议自动生成

具体自动生成的步骤如下
在这里插入图片描述
将其打勾即可
之后只要继承了Serializable这个接口,就可以通过快捷键自动生成

3. 对象-代码实战

先创建一个实体类

public class Student implements Serializable {

    private int no;
    private String name;

    public Student() {
    }

    public Student(int no, String name) {
        this.no = no;
        this.name = name;
    }

    public int getNo() {
        return no;
    }

    public void setNo(int no) {
        this.no = no;
    }

    public String getName() {
        return name;
    }

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

    @Override
    public String toString() {
        return "Student{" +
                "no=" + no +
                ", name='" + name + '\'' +
                '}';
    }


}

3.1 序列化

通过序列其对象

public class ObjectOutputStreamTest01 {
    public static void main(String[] args) throws Exception{
        // 创建java对象
        Student s = new Student(1111, "zhangsan");
        // 序列化
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("students"));

        // 序列化对象
        oos.writeObject(s);

        // 刷新
        oos.flush();
        // 关闭
        oos.close();
    }
}

查看其文件目录的时候
会生成一个students的文件
文件内部大致如下
在这里插入图片描述
这种情况类似买东西将其放置于购物车,断网断电的时候,数据被保存到此处,也就是序列化。反序列化就是将其恢复出来,从而在页面中可以看到信息还保存着

3.2 反序列化

具体代码为

import java.io.FileInputStream;
import java.io.ObjectInputStream;

/*
反序列化
 */
public class ObjectInputStreamTest01 {
    public static void main(String[] args) throws Exception{
        ObjectInputStream ois = new ObjectInputStream(new FileInputStream("students"));
        // 开始反序列化,读
        Object obj = ois.readObject();
        // 反序列化回来是一个学生对象,所以会调用学生对象的toString方法。
        System.out.println(obj);
        ois.close();
    }
}

通过反序列化的输出,在终端中输出页面显示为
在这里插入图片描述

4. 集合-代码实战

一般不可以直接传输多个对象进行序列化或者反序列化
应该将其多个对象放到集合中去

transient关键字表示游离的,不参与序列化。
将其添加到属性中,不参与其序列化,最后生成的反序列化的值是null

java 的transient关键字为我们提供了便利,你只需要实现Serilizable接口,将不需要序列化的属性前添加关键字transient,序列化对象的时候,这个属性就不会序列化到指定的目的地中

先创建一个实体类对象

import java.io.Serializable;

public class User implements Serializable {
    private int no;
    // transient关键字表示游离的,不参与序列化。
    private transient String name; // name不参与序列化操作!

    public User() {
    }

    public User(int no, String name) {
        this.no = no;
        this.name = name;
    }

    @Override
    public String toString() {
        return "User{" +
                "no=" + no +
                ", name='" + name + '\'' +
                '}';
    }

    public int getNo() {
        return no;
    }

    public void setNo(int no) {
        this.no = no;
    }

    public String getName() {
        return name;
    }

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

4.1 序列化

代码具体实现为

public class ObjectOutputStreamTest02 {
    public static void main(String[] args) throws Exception{
        List<User> userList = new ArrayList<>();
        userList.add(new User(1,"zhangsan"));
        userList.add(new User(2, "lisi"));
        userList.add(new User(3, "wangwu"));
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("users"));

        // 序列化一个集合,这个集合对象中放了很多其他对象。
        oos.writeObject(userList);

        oos.flush();
        oos.close();
    }
}

4.2 反序列化

/*
反序列化集合
 */
public class ObjectInputStreamTest02 {
    public static void main(String[] args) throws Exception{
        ObjectInputStream ois = new ObjectInputStream(new FileInputStream("users"));
        //Object obj = ois.readObject();
        //System.out.println(obj instanceof List);
        List<User> userList = (List<User>)ois.readObject();
        for(User user : userList){
            System.out.println(user);
        }
        ois.close();
    }
}
  • 5
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
很高兴帮助你解答问题。以下是Java redis序列化反序列化的示例代码: Redis序列化对象: ``` import org.springframework.data.redis.serializer.RedisSerializer; import org.springframework.data.redis.serializer.SerializationException; import org.springframework.util.Assert; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; public class RedisObjectSerializer implements RedisSerializer<Object> { private static final byte[] EMPTY_BYTE_ARRAY = new byte[0]; public RedisObjectSerializer() { super(); } public byte[] serialize(Object obj) throws SerializationException { if (obj == null) { return EMPTY_BYTE_ARRAY; } try { ByteArrayOutputStream byteStream = new ByteArrayOutputStream(128); ObjectOutputStream objectOutputStream = new ObjectOutputStream(byteStream); objectOutputStream.writeObject(obj); objectOutputStream.flush(); return byteStream.toByteArray(); } catch (Throwable ex) { throw new SerializationException("Failed to serialize", ex); } } public Object deserialize(byte[] bytes) throws SerializationException { if (isEmpty(bytes)) { return null; } try { ObjectInputStream objectInputStream = new ObjectInputStream(new ByteArrayInputStream(bytes)); return objectInputStream.readObject(); } catch (Throwable ex) { throw new SerializationException("Failed to deserialize", ex); } } private boolean isEmpty(byte[] data) { return (data == null || data.length == 0); } } ``` Redis反序列化对象: ``` import org.springframework.data.redis.serializer.RedisSerializer; import org.springframework.data.redis.serializer.SerializationException; import org.springframework.util.Assert; import java.nio.charset.Charset; public class RedisStringSerializer implements RedisSerializer<String> { private final Charset charset; public RedisStringSerializer() { this(Charset.forName("UTF8")); } public RedisStringSerializer(Charset charset) { Assert.notNull(charset, "Charset must not be null!"); this.charset = charset; } public String deserialize(byte[] bytes) { return (bytes == null ? null : new String(bytes, charset)); } public byte[] serialize(String string) throws SerializationException { return (string == null ? null : string.getBytes(charset)); } } ``` 请注意,这里使用了Spring Data Redis库,并实现了RedisSerializer接口。您可以根据需要使用其他Redis客户端库和其他序列化器。希望这可以帮助到您!

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

码农研究僧

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

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

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

打赏作者

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

抵扣说明:

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

余额充值