Java之Serializable接口实现序列化和反序列化实例以及部分序列化的四种方法

Serializable接口实现序列化和反序列化实例


首先需要明确的概念:

  1. 序列化:将数据结构或对象转换成二进制字节流的过程
  2. 反序列化:将在序列化过程中所生成的二进制字节流的过程转换成数据结构或者对象的过程
  3. 持久化:将数据写入文件中长久保存的过程称之为持久化
  4. 序列化主要目的:是通过网络传输对象或者说是将对象存储到文件系统、数据库、内存中。
  • 序列化反序列化实例:

1.创建一个学生类实现Serializable接口:

public class Student implements Serializable {
    private String stuNum;
    private String stuName;
    //学生拥有多个老师 用集合装
    private List<String> teacherList;

    public Student() {
    }

    public Student(String stuNum, String stuName, List<String> teacherList) {
        this.stuNum = stuNum;
        this.stuName = stuName;
        this.teacherList = teacherList;
    }

    @Override
    public String toString() {
        return "Student{" +
                "stuNum='" + stuNum + '\'' +
                ", stuName='" + stuName + '\'' +
                ", teacherList=" + teacherList +
                '}';
    }

    public String getStuNum() {
        return stuNum;
    }

    public void setStuNum(String stuNum) {
        this.stuNum = stuNum;
    }

    public String getStuName() {
        return stuName;
    }

    public void setStuName(String stuName) {
        this.stuName = stuName;
    }

    public List<String> getTeacherList() {
        return teacherList;
    }

    public void setTeacherList(List<String> teacherList) {
        this.teacherList = teacherList;
    }
}

2.创建SerializableUtil工具类实现序列化反序列化方法

public class SerializableUtil {
    /**
     * 将对象序列化到指定文件中
     * @param obj
     * @param fileName
     * @throws IOException
     */
    public static void mySerialize(Object obj, String fileName) throws IOException {
        OutputStream out = new FileOutputStream(fileName);
        //对象序列化反序列化的流
        ObjectOutputStream objOut = new ObjectOutputStream(out);
        //将指定对象写入ObjectOutStream
        objOut.writeObject(obj);
        //关闭流
        objOut.close();
    }

    /**
     * 从指定文件中反序列化对象
     * @param fileName
     * @return
     * @throws IOException
     * @throws ClassNotFoundException
     */
    public static Object myDeserialize(String fileName) throws IOException, ClassNotFoundException {
        InputStream in = new FileInputStream(fileName);
        ObjectInputStream objIn = new ObjectInputStream(in);
        Object obj = objIn.readObject();
        return obj;
    }
}

3.SerializableTest测试类,测试序列化反序列化实现

public class SerializableTest {

    public static void main(String[] args) {
        List<String> teacherList = new ArrayList<>();
        teacherList.add("王老师");
        teacherList.add("张老师");
        Student stu1 = new Student("1001", "张三", teacherList);
        System.out.println("原始对象:" + stu1);
        String fileName = "stu01.txt";
        try {
            //对象序列化
            SerializableUtil.mySerialize(stu1, fileName);
            System.out.println("序列化原始对象完成!OK!");
            //对象的反序列化
            Object obj = SerializableUtil.myDeserialize(fileName);
            //确定obj是Student类型
            if (obj instanceof Student) {
                Student stuNew = (Student) obj;
                System.out.println("反序列化之后的对象:" + stuNew);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

4.测试结果:
生成stu01.txt文件,将数据结构成功转为二进制字节流存入文件中:
在这里插入图片描述

在这里插入图片描述


部分属性序列化和反序列化的四种方法

Transient关键字修饰

	private transient String stuNum;
    private transient String stuName;
    //学生拥有多个老师 用集合装
    private List<String> teacherList;

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

这里的stuNumStuName反序列化后就是默认值null,因为经过了transient修饰,阻止了属性的实例化,因此反序列结果中是null

对于不想进行序列化的变量,可以使用 transient 关键字修饰。
transient 关键字的作用是:让实例中的属性不再序列化;当对象被反序列化时,被 transient 修饰的属性不会被持久化和恢复。需要注意的一些点:

  • transient关键字只能修饰属性,不能修饰方法和类。
  • transient关键字修饰的属性变量,在经过反序列化之后会是相应类型的默认值,例如String类型反序列化后就是是nullint类型反序列化后就是0

Static关键字修饰

测试:

	private String stuNum;
    private static String stuName;
    //学生拥有多个老师 用集合装
    private List<String> teacherList;
public static void main(String[] args) {
        List<String> teacherList = new ArrayList<>();
        teacherList.add("王老师");
        teacherList.add("张老师");
        Student stu1 = new Student("1001", "张三", teacherList);
        System.out.println("原始对象:" + stu1);
        String fileName = "stu01.txt";
        try {
            //对象序列化
            SerializableUtil.mySerialize(stu1, fileName);
            System.out.println("序列化原始对象完成!OK!");
            stu1.setStuName("张三他爹");
            //对象的反序列化
            Object obj = SerializableUtil.myDeserialize(fileName);
            //确定obj是Student类型
            if (obj instanceof Student) {
                Student stuNew = (Student) obj;
                System.out.println("反序列化之后的对象:" + stuNew);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

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

这里通过用static关键字修饰,直接在完成初始序列化后set.stuName可以改变学生名字,因为static修改的属性变量不属于任何对象,是静态的,因此无论使用transient与否,都不会被序列化。


修改默认方法writeObject和readObject

Serializable接口源码中有这样的说明
在这里插入图片描述

大概翻译一下writeObjectreadObject
在这里插入图片描述
在这里插入图片描述

测试:

public class Student implements Serializable {
	private String stuNum;
	private String stuName;
	private List<String> teacherList;
	
	private void writeObject(ObjectOutputStream objOut) throws IOException {
	System.out.println("writeObject-----------");
	objOut.writeObject(stuNum);
	objOut.writeObject(stuName);
	}
	
	private void readObject(ObjectInputStream objIn) throws IOException,
	ClassNotFoundException {
	System.out.println("readObject-----------");
	stuNum= (String) objIn.readObject();
	stuName= (String) objIn.readObject();
	}
}

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

因为在writeObjectreadObject方法中,只序列化和反序列化了stuNumstuName,因此teacherList是没有序列化反序列的,所以是String的默认值null


Externalizable实现序列化反序列化和部分序列化

Externalizable继承自Serializable,使用Externalizable接口需要实现readExternal方法和writeExternal方法来实现序列化和反序列化。

  • 源码部分
    在这里插入图片描述
  • 代码实现
private String stuNum;
    private String stuName;
    //学生拥有多个老师 用集合装
    private List<String> teacherList;

    @Override
    public void writeExternal(ObjectOutput out) throws IOException {
        out.writeObject(stuNum);
        out.writeObject(stuName);
    }
    
    @Override
    public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
        stuNum= (String) in.readObject();
        stuName= (String) in.readObject();
    }

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

同样是因为writeExternal()readExternal()只序列化和反序列化了stuNumstuName,因此teacherList是没有序列化反序列的,所以是String的默认值null

Serializable 与 Externalizable

区别SerializableExternalizable
实现复杂度实现简单,Java对其有内建支持实现复杂,由开发人员自己完成
执行效率所有对象由Java统一保存,性能开发人员决定哪个对象保存,可能造成速度提升
保存信息保存时占用空间大部分存储,可能造成空间减少
使用频率偏低
  • 7
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

12点前就睡

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

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

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

打赏作者

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

抵扣说明:

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

余额充值