Java小白的学习笔记(六)——对象的序列化与反序列化
对象的序列化,就是将Object对象转化为byte序列,反之叫对象的反序列化
序列化流(ObjectOutputStream)是过滤流——>writeObject方法
反序列化流(ObjectInputStream)——>readObject方法
序列化接口(Serializable)
对象必须实现序列化接口,才能进行序列化,否则会出现异常
但是这个接口没有任何方法,只是一个标准
下面我们来做一个小的测试
首先创建一个学生类并实现序列化接口
然后在测试类中测试
public static void main(String[] args) throws IOException{
//序列化操作
ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream("e:\\obj.dat"));
Student student = new Student("张三","10001",20);
objectOutputStream.writeObject(student);
objectOutputStream.flush();
objectOutputStream.close();
}
看一下结果:
好的,已经存进来了。
下面我们尽心反序列化的操作:
Student类不用修改,只需要改一下测试类:
public static void main(String[] args) throws IOException, ClassNotFoundException{
/*
//序列化操作
ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream("e:\\obj.dat"));
Student student = new Student("张三","10001",20);
objectOutputStream.writeObject(student);
objectOutputStream.flush();
objectOutputStream.close();
*/
//反序列化
ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream("e:\\obj.dat"));
Student stu = (Student) objectInputStream.readObject();
System.out.println(stu);
objectInputStream.close();
}
看一下打印结果:
完美~
下面来介绍一个关键字:transient
被transient修饰的元素,不会进行JVM默认的序列化,但是可以自己完成这个元素的序列化
那么,我们怎样完成他的序列化呢?下面我们来看一下:
private void writeObject(java.io.ObjectOutputStream s)
throws java.io.IOException{
s.defaultWriteObject();//把jvm能默认序列化的元素进行序列化操作
s.writeInt(age);//自己完成age的序列化操作,这里因为age是int类型,所以writeInt,如果是其他类型,可以直接writeObject.
}
private void readObject(java.io.ObjectInputStream s)
throws java.io.IOException, ClassNotFoundException {
s.defaultReadObject();//把jvm能默认反序列化的元素进行反序列化操作
this.age = s.readInt();//自己完成age的反序列化操作,这里因为age是int类型,所以readInt,如果是其他类型,可以直接readObject.
}
就是这两个方法,这两个方法在ArrayList的源码中可以找到,复制过来修改就可以了
接下来我们来看一下序列化中子父类构造函数的调用问题
首先我们要明白一点:一个类实现了序列化接口(Serializable),其子类都可以进行序列化
其次:序列化过程中,是会递归调用父类的构造函数的
在一个:对子类对象进行反序列化操作时,如果其父类没有实现序列化接口,那么其父类的构造函数会被调用,如果实现了序列化接口,是不会被调用的。
好了,这次笔记就到这了。满足o( ̄︶ ̄)o。