Java 中的序列化和反序列化

序列化是指将对象的状态转换为字节流的过程,以便可以在网络上传输或永久保存到磁盘中。反序列化则是将序列化的字节流恢复为对象的过程。在 Java 中,序列化和反序列化是通过 ObjectInputStream 和 ObjectOutputStream 实现的。

为什么我们需要序列化?

在现代应用程序中,对象的状态通常需要在不同的系统、进程和线程之间进行传递。例如,在分布式系统中,对象可能需要在不同的服务器之间进行传递。另一个例子是在缓存中存储对象时,对象需要序列化以便可以保存到磁盘中。

Java 中的序列化

Java 提供了一个 Serializable 接口来实现序列化。Serializable 接口只是一个标记接口,没有任何方法,它只是告诉编译器,这个类是可序列化的。如果一个类实现了 Serializable 接口,则该类的对象可以被序列化。例如:

public class Person implements Serializable {
  private static final long serialVersionUID = 1L;
  private String name;
  private int age;

  public Person(String name, int age) {
    this.name = name;
    this.age = age;
  }

  public String getName() {
    return name;
  }

  public int getAge() {
    return age;
  }
}

在上面的例子中,Person 类实现了 Serializable 接口,表示该类可以被序列化。

Java 中的反序列化

Java 中的反序列化是将序列化的字节流恢复为对象的过程。可以使用 ObjectInputStream 类实现反序列化。例如:

try {
  FileInputStream fileIn = new FileInputStream("person.ser");
  ObjectInputStream in = new ObjectInputStream(fileIn);
  Person person = (Person) in.readObject();
  in.close();
  fileIn.close();
} catch (IOException i) {
  i.printStackTrace();
} catch (ClassNotFoundException c) {
  c.printStackTrace();
}

在上面的例子中,我们首先创建了一个 FileInputStream 对象,然后将其传递给 ObjectInputStream 构造函数来创建 ObjectInputStream 对象。之后,我们使用 readObject() 方法从 ObjectInputStream 中读取对象。最后,我们关闭 ObjectInputStream 和 FileInputStream。

需要注意的是,当你进行反序列化时,序列化的类必须存在。如果序列化的类不存在,则会抛出 ClassNotFoundException 异常。

Java 中的序列化和反序列化是非常重要的,它允许对象在不同的系统、进程和线程之间进行传递。同时,Java 还提供了一些方法来控制对象的序列化和反序列化行为,例如 transient、writeObject() 和 readObject() 等方法。由于序列化和反序列化对性能的影响较大,因此在适当的情况下应该避免使用它们。