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.