自学用13.3序列化

2019-07-23  本文已影响0人  后方积水

序列化:运行中的对象->字符序列(可以储存的字符数据)

反序列化:字符序列->运行中的对象

    功能可以拿来用的地方很多,比如保存某一时刻对象的数据,然后可以用来分析或者快速启动、把数据保存到数据库、程序之间传递数据也可以。

    java.io.ObjectOutputStream 类是用来实现序列化的一种,将Java对象的原始数据类型写出到文件,实现对象的持久存储。

构造方法:public ObjectOutputStream(OutputStream out): 创建一个指定OutputStream的ObjectOutputStream。

1.一个对象要想序列化,必须满足两个条件:

该类必须实现java.io.Serializable 接口,Serializable 是一个标记接口,不实现此接口的类将不会使任何状态序列化或反序列化,会抛出NotSerializableException 。

该类的所有属性必须是可序列化的。如果有一个属性不需要可序列化的,则该属性必须注明是瞬态的,使用transient 关键字修饰。

2.写出对象方法

public final void writeObject (Object obj) : 将指定的对象写出。

public class SerializeDemo{

  public static void main(String [] args)  {

    Employee e = new Employee();

    e.name = "zhangsan";

    e.address = "beiqinglu";

    e.age = 20;

    try {

      // 创建序列化流对象

          ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("employee.txt"));

        // 写出对象

out.writeObject(e);

        // 释放资源

        out.close();

       //  fileOut.close(); 这句教案上有但按代码来看应该是错的

        System.out.println("Serialized data is saved"); // 姓名,地址被序列化,年龄没有被序列化。

        } catch(IOException i)  {

            i.printStackTrace();

        }

     }

}

输出结果:

Serialized data is saved

3.3用 ObjectInputStream类来恢复

方式一


如果能找到一个对象的class文件,我们可以进行反序列化操作,调用ObjectInputStream读取对象的方法:

构造方法:public ObjectInputStream(InputStream in): 创建一个指定InputStream的ObjectInputStream。

public final Object readObject () : 读取一个对象。

public class DeserializeDemo {

  public static void main(String [] args)  {

        Employee e = null;

        try {

            // 创建反序列化流

            FileInputStream fileIn = new FileInputStream("employee.txt");

            ObjectInputStream in = new ObjectInputStream(fileIn);

            // 读取一个对象

            e = (Employee) in.readObject();

            // 释放资源

            in.close();

            fileIn.close();

        }catch(IOException i) {

            // 捕获其他异常

            i.printStackTrace();

            return;

        }catch(ClassNotFoundException c)  {

        // 捕获类找不到异常

            System.out.println("Employee class not found");

            c.printStackTrace();

            return;

        }

        // 无异常,直接打印输出

        System.out.println("Name: " + e.name); // zhangsan

        System.out.println("Address: " + e.address); // beiqinglu

        System.out.println("age: " + e.age); // 0

    }

}

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

方式二

当JVM反序列化对象时,能找到class文件,但是class文件在序列化对象之后发生了修改,那么反序列化操作也会失败,抛出一个InvalidClassException异常。发生这个异常的原因如下:

该类的序列版本号与从流中读取的类描述符的版本号不匹配

该类包含未知数据类型

该类没有可访问的无参数构造方法

Serializable 接口给需要序列化的类,提供了一个序列版本号。serialVersionUID 该版本号的目的在于验证序列化的对象和对应类是否版本匹配。

所以可以主动给序列一个版本号,这样新增了成员之后也能读取序列化的对象,新增的对象为默认值。

public class Employee implements java.io.Serializable {

    // 加入序列版本号

    private static final long serialVersionUID = 1L;

    public String name;

    public String address;

    // 添加新的属性 ,重新编译, 可以反序列化,该属性赋为默认值.

   比如:  public int eid;

    public void addressCheck() {

        System.out.println("Address  check : " + name + " -- " + address);

    }

}

上一篇下一篇

猜你喜欢

热点阅读