美文网首页
自学用13.3序列化

自学用13.3序列化

作者: 后方积水 | 来源:发表于2019-07-23 21:26 被阅读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);

        }

    }

    相关文章

      网友评论

          本文标题:自学用13.3序列化

          本文链接:https://www.haomeiwen.com/subject/floalctx.html