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.