序列化和反序列化原理解析及ObjectOutputStream出现EOF异常分析解决

为何要序列化(序列化协议)?持久化和网络传输(挤压空间,减少网络传输无效的消耗;二进制传输方便) 应用场景:dubbo+dto+rpc调用
代替品:json,pb
Java序列化机制会根据编译的class自动生成一个serialVersionUID 作为序列化版本比较,这种情况下,只有同一次编译生成相同的serialVersionUID的class文件生成的对象才能够反序列化。

1. 实体对象没有实现序列化接口

@Data
@Accessors(chain = true)
public class User{

    private String id;
    private String name;
    private String sex;
    private String age;
    private String email;
    private String addr;
}
/*
    * 序列化和反序列化
    * */
    @Test
    public void test34() {
        User user = new User().setId(UUID.randomUUID().toString())
                .setName("zhangsan")
                .setAge("23")
                .setEmail("xxx@qq.com")
                .setSex("男")
                .setAddr("广东省");

        try(FileOutputStream fileOutputStream = new FileOutputStream("D:\\对象序列化.txt");
            ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);
            FileInputStream fileInputStream = new FileInputStream("D:\\对象序列化.txt");
            ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);)
        {
            //序列化
            objectOutputStream.writeObject(user);
            //反序列化
            objectInputStream.readObject();
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException | ClassNotFoundException e) {
            e.printStackTrace();
        }
    }

结果:直接报错
在这里插入图片描述
分析:jdk实现的一种对象版本一致性保证机制,默认不实现序列化接口是不能保证对象版本一致

2. 实体对象实现序列化接口,但没有指定序列化版本

在这个实验过程中遇到点bug,这个点也是比较重要的点,里面的细节也是需要我们get到 。

@Data
@Accessors(chain = true)
public class User implements Serializable{

    private String id;
    private String name;
    private String sex;
    private String age;
    private String email;
    private String addr;
}
 @Test
    public void test34() {
        User user = new User();
        user.setId(UUID.randomUUID().toString());
        try(FileOutputStream fileOutputStream = new FileOutputStream("D:\\对象序列化.txt");
            ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);
            FileInputStream fileInputStream = new FileInputStream("D:\\对象序列化.txt");
            ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);)
        {
            //序列化
            //objectOutputStream.writeObject(user);
            //反序列化
            System.out.println((User)objectInputStream.readObject());
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException | ClassNotFoundException e ) {
            e.printStackTrace();
        }
    }

具体操作:先注释反序列化,运行序列化;再注释序列化,运行反序列化
结果:
在这里插入图片描述
显然这个结果并不是我们此次希望的结果。
什么原因呢?反序列化的时候没有注释

FileOutputStream fileOutputStream = new FileOutputStream("D:\\对象序列化.txt");
            ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);

导致ObjectOutputStream初始化的时候就会输出四个字节的头信息,覆盖掉序列化时候的信息,导致表面上我们反序列化的时候即使看似没有写入信息,但是仍然没有读到东西,现象:
在这里插入图片描述
解决:反序列化的时候注释掉(或者将写读分开封装到单独的方法)

FileOutputStream fileOutputStream = new FileOutputStream("D:\\对象序列化.txt");
            ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);

现在回归主题:
先序列化

    /*
    * 序列化和反序列化
    * */
    @Test
    public void test34() {
        User user = new User();
        user.setId(UUID.randomUUID().toString());
        try(FileOutputStream fileOutputStream = new FileOutputStream("D:\\对象序列化.txt");
            ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);
            FileInputStream fileInputStream = new FileInputStream("D:\\对象序列化.txt");
            ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);)
        {
            //序列化
            objectOutputStream.writeObject(user);
            //反序列化
           //System.out.println((User)objectInputStream.readObject());
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e ) {
            e.printStackTrace();
        }
    }

修改类信息:
在这里插入图片描述
反序列化:

 /*
    * 序列化和反序列化
    * */
    @Test
    public void test34() {
        User user = new User();
        user.setId(UUID.randomUUID().toString());
        try(/*FileOutputStream fileOutputStream = new FileOutputStream("D:\\对象序列化.txt");
            ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);*/
            FileInputStream fileInputStream = new FileInputStream("D:\\对象序列化.txt");
            ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);)
        {
            //序列化
            //objectOutputStream.writeObject(user);
            //反序列化
           System.out.println((User)objectInputStream.readObject());
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException | ClassNotFoundException e ) {
            e.printStackTrace();
        }
    }

结果:
在这里插入图片描述

java.io.InvalidClassException: test.pojo.User; local class incompatible: stream classdesc serialVersionUID = -6827564090699398067, local class serialVersionUID = -8230425444630182070

果然,一旦类的信息修改,会自动修改serialVersionUID,就跟之前序列化磁盘的对象对应的类的serialVersionUID 不一样了。
那么我们如何规避这种一致性机制呢,按我们开发人员所认为的一致来走呢?下面就到实体类中定一个serialVersionUID了。

3. 实体对象实现序列化接口,指定序列化版本

@Data
@Accessors(chain = true)
public class User implements Serializable{
    private static final long serialVersionUID = 1L;
    private String id;
    private String name;
    private String sex;
    private String age;
    private String email;
    private String addr;
    private String nickName;
}

重复2中的测试
结果:
在这里插入图片描述

User(id=38adcac4-bd13-4397-95da-37f051c51157, name=null, sex=null, age=null, email=null, addr=null, nickName=null)

非常nice;当我们指定序列化版本时,序列化到磁盘的对象会保存序列化id,反序列化的时候就会将磁盘对象中的序列化id和类指定的序列化id,一致就说明版本一致,就可以实现反序列化赋值了。

4.ObjectOutputStream初始化过程:

ObjectOutputStream初始化的时候会输出4个字节的头信息,也就是上面txt文本开头的两个“口口”,
在这里插入图片描述
单独使用ObjectOutputStream:

 @Test
    public void test35() throws IOException {
        FileOutputStream fileOutputStream = new FileOutputStream("D:\\对象序列化.txt");
            ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);
        objectOutputStream.close();
        fileOutputStream.close();
    }

在这里插入图片描述

源码分析
在这里插入图片描述

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

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
以下是Java的实现代码: ```java import java.io.*; public class Product implements Serializable { private int ID; private String name; private String property; private double price; public Product(int ID, String name, String property, double price) { this.ID = ID; this.name = name; this.property = property; this.price = price; } public int getID() { return ID; } public void setID(int ID) { this.ID = ID; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getProperty() { return property; } public void setProperty(String property) { this.property = property; } public double getPrice() { return price; } public void setPrice(double price) { this.price = price; } @Override public String toString() { return "ID: " + ID + ", Name: " + name + ", Property: " + property + ", Price: " + price; } } public class Main { public static void main(String[] args) { Product product1 = new Product(1, "iPhone", "Smartphone", 999); Product product2 = new Product(2, "iPad", "Tablet", 499); Product product3 = new Product(3, "MacBook", "Laptop", 1299); Product product4 = new Product(4, "iMac", "Desktop", 1499); Product product5 = new Product(5, "AirPods", "Wireless Earbuds", 199); try { FileOutputStream fileOut = new FileOutputStream("D:\\序列化.txt"); ObjectOutputStream out = new ObjectOutputStream(fileOut); out.writeObject(product1); out.writeObject(product2); out.writeObject(product3); out.writeObject(product4); out.writeObject(product5); out.close(); fileOut.close(); System.out.println("Serialized data is saved in D:\\序列化.txt"); } catch (IOException e) { e.printStackTrace(); } try { FileInputStream fileIn = new FileInputStream("D:\\序列化.txt"); ObjectInputStream in = new ObjectInputStream(fileIn); while (true) { try { Product product = (Product) in.readObject(); System.out.println(product); } catch (EOFException e) { break; } } in.close(); fileIn.close(); } catch (IOException e) { e.printStackTrace(); } catch (ClassNotFoundException e) { e.printStackTrace(); } } } ``` 这个程序创建了一个名为Product的类,这个类有4个属性:ID、name、property、price。在初始化函数中,这些属性会被赋予相应的值。程序创建了5个Product对象,并将它们序列化到名为“序列化.txt”的文件中。接下来,程序从文件中反序列化对象,并将它们输出到控制台中。 需要注意的是,为了让对象可以被序列化反序列化,Product类需要实现Serializable接口。此外,在序列化反序列化过程中,需要用到ObjectOutputStreamObjectInputStream类。序列化时,需要先创建一个ObjectOutputStream对象,然后使用它的writeObject()方法将对象写入文件;反序列化时,需要先创建一个ObjectInputStream对象,然后使用它的readObject()方法从文件中读取对象。在读取对象时,需要用try-catch语句捕获EOFException异常,以便在读取完所有对象后退出循环。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Fire king

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

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

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

打赏作者

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

抵扣说明:

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

余额充值