Почему статическое переходное поле сериализуется?
Я читал, что статические поля не сериализуются, но после тестирования я увидел, что это не так.
Статический модификатор даже переопределяет переходный модификатор и делает поле сериализуемым.
Я пишу один пример из книги, которая показывает, что статическое переходное поле сериализовано.
import java.io.*;
class USPresident implements Serializable {
private static final long serialVersionUID = 1L;
@Override
public String toString() {
return "US President [name=" + name
+ ", period=" + period + ", term=" + term + "]";
}
public USPresident(String name, String period, String term) {
this.name = name;
this.period = period;
this.term = term;
}
private String name;
private String period;
private static transient String term;
}
class TransientSerialization {
public static void main(String[] args) {
USPresident usPresident = new USPresident("Barack Obama", "2009 to --", "56th term");
System.out.println(usPresident);
try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("USPresident.data"))) {
oos.writeObject(usPresident);
} catch (IOException ioe) {
// ignore
}
try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream("USPresident.data"))) {
Object obj = ois.readObject();
if (obj != null && obj instanceof USPresident) {
USPresident presidentOfUS = (USPresident) obj;
System.out.println(presidentOfUS);
}
} catch (IOException ioe) {
// ignore
} catch (ClassNotFoundException e) {
// ignore
}
}
}
Неправильно ли общее представление о том, что статические поля не сериализуются? Это просто рекомендация? Почему модификатор переходного процесса не работает со статическим?
примечание: я понимаю, что инициализация статического поля в конструкторе - это странный код, но компилятор позволил мне это сделать, и это просто для того, чтобы понять сериализацию статических полей.
1 ответ
Это не имеет ничего общего с сериализацией, но из-за того, что вы устанавливаете статическое поле при создании переменной usPresident. Это устанавливает поле для класса этой JVM. Попробуйте прочитать в сериализованном президенте в другой программе и убедитесь, что переходное поле не сериализовано.
В качестве отступления: не игнорируйте ваши исключения.
Например, рефакторинг, ваш код может выглядеть так:
class USPresident implements Serializable {
private static final long serialVersionUID = 1L;
@Override
public String toString() {
return "US President [name=" + name + ", period=" + period + ", term="
+ term + "]";
}
public USPresident(String name, String period, String term) {
this.name = name;
this.period = period;
this.term = term;
}
private String name;
private String period;
private static transient String term;
}
class TransientSerialization {
public static void main(String[] args) {
serializePresident();
deserializePresident();
}
private static void deserializePresident() {
ObjectInputStream ois = null;
try {
ois = new ObjectInputStream(new FileInputStream(
"USPresident.data"));
Object obj = ois.readObject();
if (obj != null && obj instanceof USPresident) {
USPresident presidentOfUS = (USPresident) obj;
System.out.println(presidentOfUS);
}
} catch (IOException ioe) {
ioe.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
} finally {
if (ois != null) {
try {
ois.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
private static void serializePresident() {
USPresident usPresident = new USPresident("Barack Obama", "2009 to --",
"56th term");
System.out.println(usPresident);
ObjectOutputStream oos = null;
try {
oos = new ObjectOutputStream(new FileOutputStream("USPresident.data"));
oos.writeObject(usPresident);
oos.close();
} catch (IOException ioe) {
ioe.printStackTrace();
} finally {
if (oos != null) {
try {
oos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
Во второй раз, когда вы запустите его, измените метод main на:
public static void main(String[] args) {
// serializePresident();
deserializePresident();
}
И посмотри, что подходит.
Для меня первый запуск возвращает:
US President [name=Barack Obama, period=2009 to --, term=56th term]
US President [name=Barack Obama, period=2009 to --, term=56th term]
и второй прогон возвращает:
US President [name=Barack Obama, period=2009 to --, term=null]