用于序列化单例的瞬态



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?