Effective Java -要维护单例保证,您必须将所有实例字段声明为transient,并提供一个'readResolve'方法。我们在这里通过声明暂态字段来实现什么?下面是一个示例:
....
....
public final class MySingleton implements Serializable{
private int state;
private MySingleton() { state =15; }
private static final MySingleton INSTANCE = new MySingleton();
public static MySingleton getInstance() { return INSTANCE; }
public int getState(){return state;}
public void setState(int val){state=val;}
private Object readResolve() throws ObjectStreamException {
return INSTANCE;
}
public static void main(String[] args) {
MySingleton c = null;
try {
c=MySingleton.getInstance();
c.setState(25);
FileOutputStream fs = new FileOutputStream("testSer.ser");
ObjectOutputStream os = new ObjectOutputStream(fs);
os.writeObject(c);
os.close();
} catch (Exception e) {
e.printStackTrace();
}
try {
FileInputStream fis = new FileInputStream("testSer.ser");
ObjectInputStream ois = new ObjectInputStream(fis);
c = (MySingleton) ois.readObject();
ois.close();
System.out.println("after deser: contained data is " + c.getState());
} catch (Exception e) {
e.printStackTrace();
}
}
}
无论我是否将'state'变量声明为瞬态变量,我都会得到c.getState()输出为25。我错过什么了吗?
将属性设置为瞬态的好处是不必序列化状态。序列化它是不必要的,因为它无论如何都会被readResolve()方法丢弃。
如果状态包含在int类型中,则无关紧要。但如果状态是一个复杂的对象图,则会产生显著的性能差异。当然,如果状态是不可序列化的,你就没有其他选择了。
也就是说,序列化单例是有问题的。
无论我是否将'state'变量声明为瞬态变量,我都会得到c.getState()输出为25。我错过什么了吗?
替换反序列化对象,因为readResolve()返回MySingleton的私有静态实例,从而覆盖反序列化返回的实例。
try {
c=MySingleton.getInstance();
c.setState(25);
FileOutputStream fs = new FileOutputStream("testSer.ser");
ObjectOutputStream os = new ObjectOutputStream(fs);
os.writeObject(c);
os.close();
}
你得到了25,因为你在对象中设置了那个值。参考上面的代码。我想你以为答案是15?