美文网首页各种IOJava IO专题程序员
Java IO之序列化和反序列化

Java IO之序列化和反序列化

作者: 第四单元 | 来源:发表于2019-01-15 16:17 被阅读39次

    一.基本知识

    Java语言支持一种称为对象序列化和反序列的机制,它可以将任意对象写出到流中,并在之后将其读回,恢复出一个新的对象。

    1.1如何序列化

    首先,对于希望序列化类,都必须实现Serializable接口,该接口没有任何方法定义,只是一个标识。如果没有实现该接口,ObjectOutputStream.writeObject实例方法则会抛出NotSerializableException异常。

    首先需要打开一个ObjectOutputStream对象:

    ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("data.dat"));
    

    使用out.writeObject(Object)将对象序列化
    对于基本数据类型,使用out.writeInt()\out.writeDouble()\out.writeFloat()等方法。

    对象被多个对象引用的情况
    如果序列化的多个对象中都有对同一个对象的引用怎么处理?

    序列化时:

    • 每个对象都管理一个序列号
    • 当第一次遇到时,保存其到数据流中
    • 当第二次遇到时,只记录“与之前保存过的序列号为x的对象相同”这个信息即可。在反序列化时执行反过来的操作。

    反序列化时:

    • 对于流中的对象,在第一次遇到其序列号时,构建它,并使用流中数据来初始化它,然后记录这个顺序号和新对象之间的关联。
    • 当遇到“与之前保存过的序列号为x的对象相同”标记时,获取与这个顺序号相关联的对象引用

    这就是序列化这个名字的来源(因为这里有个序列号).

    1.2如何反序列化

    获取一个ObjectInputStream对象:

    ObjectInputStream in = new ObjectInputStream(new FileInputStream("data.dat"));
    

    用readObject()方法以这些对象被写出时的顺序获取它们

    Employee e1 = (Employee) in.readObject();
    Employee e2 = (Employee) in.readObject();
    

    二.修改默认的序列化机制

    2.1阻止序列化某些属性

    某些数据域不需要进行序列,如只对本地方法有意义的存储文件句柄和窗口句柄的整数值。还有些域不想让它们序列化。这时可以用transient修饰这些域。

    对于不可序列化的属性,也需要用transient修饰,否则序列化时会抛出异常。

    2.2自定义序列化

    可以在类中实现一下方法:

    private void readObject(ObjectInputStream in) throws IOEeception,ClassNOtFoundException;
    
    private void writeObject(OjbectOutputStream out) throws IOException;
    

    之后,再序列化或反序列这个类的对象时就不再采用默认的规则,而是调用类的这两个方法。

    三.序列化单例

    问题:程序中有一个单例的对象。如果对其进行序列化,再反序列化,则得到了另一个对象。破坏了其单例性。

    解决方法是在类中定义readResolve方法:

    public final class Singleton implements Serializable {
      private Singleton() {}
      private static final Singleton INSTANCE = new Singleton();
      public static Singleton getInstance() { return INSTANCE; }
      private Object readResolve() throws ObjectStreamException {
        return INSTANCE;
      }
    }
    

    这样,当反序列化时,就会调用这个readResolve方法返回我们制定好的对象。

    四.版本管理

    serialVersionUID的作用:
    序列号操作的时候系统会把当前类的serialVersionUID写入到序列化流输出流中,当反序列化时系统会检查流中的serialVersionUID是否和当前类的serialVersionUID一致。如果一致,就说明可以反序列化,否则readObject会抛出InvalidClassException异常。

    serialVersionUID的生成:

    • 可以不显示地指定这个常量,那么序列号化时jvm会自动生成一个版本号(根据当前类的信息,如类名、属性等)。这样每当类改变时,就不能兼容之前版本的已序列化的对象了;
    • 如果在类中人工地指定了序列号private static final long serialVersionUID。则不会再自动生成,那么不同版本的类之间会尽量进行兼容,不会抛出InvalidClassException。如果流中的属性比当前的类多,则忽略多余的。如果当前类的属性比流中的多,则多余的赋成默认值;

    五.用序列化的方式克隆对象

    可以将一个对象序列化后再反序列化,就相当于深克隆了这个对象。

    但这种方式效率比较低,不如直接显示地构建对象进行克隆快。

    相关文章

      网友评论

        本文标题:Java IO之序列化和反序列化

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