单例模式的问题:
1 AccessibleObject.setAccessible反射可以调用私有构造方法
2 反序列化会生成一个新的实例
解决:
1 防止反射调用的话可以修改私有构造函数,使其在请求创建第二个实例时抛出异常,且提供一个静态的公开的工厂函数。
2 ObjectInputStream在反序列化的时候会检测 hasReadResolveMethod,如果单例类中存在readResolve方法,则调用,所以可以在单例类中加入这个方法,防止序列化生成新的实例
// readResolve method to preserve singleton property
private Object readResolve() {
// Return the one true Elvis and let the garbage collector
// take care of the Elvis impersonator.
return INSTANCE;
}
3 一个单例的第三种方法是声明单一元素的枚举类
// Enum singleton - the preferred approach
public enum Elvis {
INSTANCE;
public void leaveTheBuilding() { ... }
}
反编译完了之后 INSTANCE就是Elvis类的一个public static final的实例,并且Constructor的newInstance方法对枚举类进行了特殊处理,另外序列化一个枚举类的对象,获取的时候调用的是继承的Enum的valueOf 方法T result = enumType.enumConstantDirectory().get(name);根据name去找存入的对象,所以不会生成多个对象。