Учебники

Java — Сериализация

Java предоставляет механизм, называемый сериализацией объекта, где объект может быть представлен в виде последовательности байтов, которая включает в себя данные объекта, а также информацию о типе объекта и типах данных, хранящихся в объекте.

После того, как сериализованный объект был записан в файл, его можно прочитать из файла и десериализовать, то есть информацию о типе и байты, которые представляют объект и его данные, можно использовать для воссоздания объекта в памяти.

Наиболее впечатляет то, что весь процесс независим от JVM, то есть объект может быть сериализован на одной платформе и десериализован на совершенно другой платформе.

Классы ObjectInputStream и ObjectOutputStream — это высокоуровневые потоки, которые содержат методы для сериализации и десериализации объекта.

Класс ObjectOutputStream содержит много методов записи для записи различных типов данных, но особенно выделяется один метод —

public final void writeObject(Object x) throws IOException

Вышеуказанный метод сериализует объект и отправляет его в выходной поток. Аналогично, класс ObjectInputStream содержит следующий метод десериализации объекта:

public final Object readObject() throws IOException, ClassNotFoundException

Этот метод извлекает следующий объект из потока и десериализует его. Возвращаемое значение — Object, поэтому вам необходимо привести его к соответствующему типу данных.

Чтобы продемонстрировать, как сериализация работает в Java, я собираюсь использовать класс Employee, который мы обсуждали в начале книги. Предположим, что у нас есть следующий класс Employee, который реализует интерфейс Serializable —

пример

public class Employee implements java.io.Serializable {
   public String name;
   public String address;
   public transient int SSN;
   public int number;
   
   public void mailCheck() {
      System.out.println("Mailing a check to " + name + " " + address);
   }
}

Обратите внимание, что для успешной сериализации класса должны быть выполнены два условия:

  • Класс должен реализовывать интерфейс java.io.Serializable.

  • Все поля в классе должны быть сериализуемыми. Если поле не сериализуемо, оно должно быть помечено как временное .

Класс должен реализовывать интерфейс java.io.Serializable.

Все поля в классе должны быть сериализуемыми. Если поле не сериализуемо, оно должно быть помечено как временное .

Если вам интересно узнать, можно ли сериализовать стандартный класс Java, проверьте документацию по этому классу. Тест прост: если класс реализует java.io.Serializable, то он сериализуем; в противном случае это не так.

Сериализация объекта

Класс ObjectOutputStream используется для сериализации объекта. Следующая программа SerializeDemo создает экземпляр объекта Employee и сериализует его в файл.

Когда выполнение программы завершено, создается файл с именем employee.ser. Программа не генерирует никаких выходных данных, но изучает код и пытается определить, что делает программа.

Примечание. При сериализации объекта в файл стандартное соглашение в Java — дать файлу расширение .ser .

пример

import java.io.*;
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();
      }
   }
}

Десериализация объекта

Следующая программа DeserializeDemo десериализует объект Employee, созданный в программе SerializeDemo. Изучите программу и попытайтесь определить ее результат —

пример

import java.io.*;
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

Вот следующие важные моменты, которые следует отметить —

Блок try / catch пытается перехватить исключение ClassNotFoundException, которое объявляется методом readObject (). Чтобы JVM могла десериализовать объект, она должна найти байт-код для класса. Если JVM не может найти класс во время десериализации объекта, она генерирует исключение ClassNotFoundException.

Обратите внимание, что возвращаемое значение readObject () приведено к ссылке Employee.

Значение поля SSN было 11122333, когда объект был сериализован, но поскольку поле является временным, это значение не было отправлено в выходной поток. Поле SSN десериализованного объекта Employee равно 0.