序列化

概述

  • Java提供了一种对象序列化的机制。用一个字节序列可以表示一个对象,该字节序列包含该对象的数据、对象的类型和对象中存储的属性等信息。字节序列化写出到文件之后,相当于文件中持久保存了一个对象的信息。反之,该字节序列还可以从文件中读取回来,重构对象,对它进行反序列化。对象的数据、对象的类型和对象中存储的数据信息,都可以用来在内存中创建对象。

ObjectOutputStream类

  • java.io.ObjectOutputStream类,将Java对象的原始数据类型写出到文件,实现对象的持久存储。
  • 构造方法
  • ObjectOutputStream(OutputStream out):创建一个写入指定的OutputStream的ObjectOutputStream
  • 构造举例,代码如下:
FileOutputStream fileOut = new FileOutputStream("a.txt");
ObjectOutputStream out = new ObjectOutputStream(fileOut);
  • 序列化操作
  • 一个对象要想序列化,必须满足两个条件:
  • 该类必须实现java.ioSerializable接口,Serializable是一个标记接口,不实现此接口的类将不会使任何状态序列化或反序列化,会抛出NotSeriableException。
  • 该类的所有属性必须是可序列化的。如果有一个属性不需要可序列化的,则该属性必须注明是瞬态的,使用transient关键字修饰。
package com.io.objectstream;

import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectOutputStream;

/*
     java.io.ObjectOutputStream extends OutputStream
     ObjectOutputStream:对象的序列化流
     作用:把对象以流的方式写入到文件中保存

     构造方法:
              ObjectOutputStream(OutputStream out)创建一个写入指定的OutputStream的ObjectOutputStream。
              参数:
                   OutputStream out :字节输出流。
     特有的成员方法:
              void writeObject(Object obj)将指定的对象写入  ObjectOutputStream。

     使用步骤:
              1.创建ObjectOutputStream对象,构造方法中传递字节输出流。
              2.使用ObjectOutputStream对象中的方法writeObject,把对象写入到文件中
              3.释放资源

     注意:
           序列化和反序列化的时候,会抛出NotSerializableException没有序列化异常
           类通过实现java.io.Serializable接口以启用其序列化功能。未实现此接口的类将无法使其任何状态序列化或反序列化。
           Serializable接口也叫标记型接口
                 要进行序列化和反序列化的类必须实现Serializable接口,就会给类添加一个标记
                 当我们进行序列化和反序列化的时候,就会检测类上是否有这个标记
                      有:就可以序列化和反序列化
                      没有:就会抛出NotSerializableException异常
 */
public class ObjectOutputstreamDemo01 {
    public static void main(String[] args) throws IOException {
        //1.创建ObjectOutputStream对象,构造方法中传递字节输出流。
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("C:\\abc\\person.txt"));
        //2.使用ObjectOutputStream对象中的方法writeObject,把对象写入到文件中
        oos.writeObject(new Person("蛇皮",19));//NotSerializableException
        //3.释放资源
        oos.close();
    }

}

ObjectInputStream类

  • ObjectInputStream反序列化流,将之前使用ObjectOutputStream序列化的原始数据恢复为对象。
  • 构造方法
  • ObjectInputStream(InputStream in):创建从指定的InputStream读取的ObjectInputStream。
  • 反序列化操作1
  • 如果能找到一个对象的class文件,我们可以进行反序列化操作,调用ObjectInputStream读取对象的方法:
    public final Object readObject():读取一个对象。
package com.io.objectstream;

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

/*
    java.io.ObjectInputStream extends InputStream
    ObjectInputStream:对象的反序列化流
    作用:把文件中保存的对象,以流的方式读取出来使用

    构造方法:
             	ObjectInputStream(InputStream in)创建从指定的InputStream读取的ObjectInputStream。
             	参数:
             	    InputStream in:字节输入流
    特有的成员方法:
               Object readObject() 从ObjectInputStream读取对象。
    使用步骤:
                1.创建ObjectInputStream对象,构造方法中传递字节输入流
                2.使用ObjectInputStream对象中的方法readObject读取保存对象中的文件
                3.释放资源
                4.使用读取处理的对象(打印)
 */
public class ObjectInputStreamDemo01 {
    public static void main(String[] args) throws IOException, ClassNotFoundException {
        // 1.创建ObjectInputStream对象,构造方法中传递字节输入流
        ObjectInputStream ois = new ObjectInputStream(new FileInputStream("C:\\abc\\person.txt"));
        //2.使用ObjectInputStream对象中的方法readObject读取保存对象中的文件
        Object o = ois.readObject();
        //3.释放资源
        ois.close();
        //4.使用读取处理的对象
        System.out.println(o);
    }
}

对于JVM可以反序列化对象,它必须是能找到class文件的类。如果找不到该类的class文件,则抛出一个ClassNotFoundException异常。

  • 反序列化操作2
  • 另外,当JVM反序列化对象时,能找到class文件,但是class文件在序列化对象之后发生了修改,那么反序列化操作也会失败,抛出一个InvalidClassException异常。发生这个异常的原因如下:
  • 该类的序列化版本号与流中读取的类描述的版本号不匹配
  • 该类包含未知数据类型
  • 该类没有可访问的无参数构造方法
  • Serializable接口给需要序列化的类,提供了一个序列化版本号。serialVersionUID该版本号的目的在于验证序列化的对象和对应类是否版本匹配。
private static final long serialVersionUID = 1L;
    private String name;
    private int age;

练习:序列化集合

  • 将存有多个自定义对象的集合序列化操作,保存到list.txt文件中,反序列化list.txt,并遍历集合,打印对象信息。
  • 代码实现:
package com.io.objectstream;

import java.io.*;
import java.util.ArrayList;

/*
     练习:序列化集合

     分析:
           1.定义一个存储Person对象的ArrayList集合
           2.往ArrayList集合中存储Person对象
           3.创建一个序列化流ObjectOutputStream对象
           4.使用ObjectOutputStream对象中的方法writeObject,对集合进行序列化
           5.创建一个反序列化ObjectInputStream对象
           6.使用ObjectInputStream对象中的方法readObject读取文件中保存的集合
           7.把Object类型的集合转换为ArrayList类型
           8.遍历ArrayList集合
           9.释放资源
 */
public class TestDemo01 {
    public static void main(String[] args) throws IOException, ClassNotFoundException {
        //1.定义一个存储Person对象的ArrayList集合
        ArrayList<Person> list = new ArrayList<>();
        //2.往ArrayList集合中存储Person对象
        list.add(new Person("唐蛇皮",21));
        list.add(new Person("菜虚困",20));
        list.add(new Person("药羊",22));
        //3.创建一个序列化流ObjectOutputStream对象
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("C:\\abc\\list.txt"));
        //4.使用ObjectOutputStream对象中的方法writeObject,对集合进行序列化
        oos.writeObject(list);
        //5.创建一个反序列化ObjectInputStream对象
        ObjectInputStream ois = new ObjectInputStream(new FileInputStream("C:\\abc\\list.txt"));
        //6.使用ObjectInputStream对象中的方法readObject读取文件中保存的集合
        Object o = ois.readObject();
        //7.把Object类型的集合转换为ArrayList类型
        ArrayList<Person> list2 = (ArrayList<Person>)o;
        //8.遍历ArrayList集合
        for (Person p: list2) {
            System.out.println(p);
        }
        //9.释放资源
        ois.close();
        oos.close();

    }
}