1. Java序列化

    Java 提供了一种对象序列化的机制。

    该机制中,一个对象可以被表示为一个字节序列,该字节序列包括该对象的数据、有关对象的类型的信息和存储在对象中数据的类型。

    序列化:将对象写入文件中

    反序列化:将对象从文件中读取出来。也就是说使用保存的对象的类型信息、对象的数据、对象中的数据类型在内存中新建该对象。

    整个过程都是 Java 虚拟机(JVM)独立的,也就是说,在一个平台上序列化的对象可以在另一个完全不同的平台上反序列化该对象。

    1. 类 ObjectInputStream包含序列化一个对象的方法。

      public final void writeObject(Object x) throws IOException
    2. 类ObjectInputStream 包含反序列化一个对象的方法

      public final Object readObject() throws IOException,ClassNotFoundException

      该方法返回值为Object,因此,需要将它转换成合适的数据类型。

  2. Java序列化演示

    1. 要被序列化的类Emloyee

      public class Employee implements java.io.Serializable
      {
         public String name;
         public String address;
         public transient int SSN;
         public int number;
      }

      一个类的对象要想序列化成功,必须满足两个条件:

      (1)该类必须实现 java.io.Serializable 接口。

      (2)该类的所有属性必须是可序列化的。如果有一个属性不是可序列化的,则该属性必须注明是短暂的。

      检验一个类的实例是否能序列,只需要查看该类有没有实现 java.io.Serializable接口。

    2. 序列化对象

      public class SerializeDemo
      {
         public static void main(String [] args)
         {
            Employee e = new Employee();
            e.name = "Reyan Ali";
            e.address = "Phokka Kuan, Ambehta Peer";
            e.SSN = 11122333;
            e.number = 101;
            try
            {
               FileOutputStream fileOut = new FileOutputStream("/tmp/employee.ser");
               ObjectOutputStream out = new ObjectOutputStream(fileOut);
               out.writeObject(e);
               out.close();
               fileOut.close();
               System.out.printf("Serialized data is saved in /tmp/employee.ser");
            }catch(IOException i)
            {
                i.printStackTrace();
            }
         }
      }

      当序列化一个对象到文件时, 按照 Java 的标准约定文件扩展名是 .ser 。

    3. 反序列化对象

      public class DeserializeDemo
      {
         public static void main(String [] args)
         {
            Employee e = null;
            try
            {
               FileInputStream fileIn = new FileInputStream("/tmp/employee.ser");
               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("Deserialized Employee...");
            System.out.println("Name: " + e.name);
            System.out.println("Address: " + e.address);
            System.out.println("SSN: " + e.SSN);
            System.out.println("Number: " + e.number);
          }
      }

      输出结果:

      Deserialized Employee...
      Name: Reyan Ali
      Address:Phokka Kuan, Ambehta Peer
      SSN: 0
      Number:101

      当对象被序列化时,属性 SSN 的值为 111222333,但是因为该属性是短暂的,该值没有被发送到输出流。所以反序列化后 Employee 对象的 SSN 属性为 0。

  3. transient