序列化和反序列化原理解析及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
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Fire king

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

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

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

打赏作者

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

抵扣说明:

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

余额充值