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.

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

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

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

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

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

Note - При сериализации объекта в файл стандартным соглашением в 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.