在上一篇文章《单例模式的终结者——setAccessible(true)》中介绍了传统单例模式的不足之处,虽然枚举类实现单例模式还没有被广泛采用,但《effective java》里面已经承认单元素枚举类是实现单例模式最好的方法了。
下面写个小demo示范一下,这是只有一个元素的枚举类,枚举类里面也可以写方法。
package go.derek;
public enum EnumSingleton {
instance;
public void doSomething(){
System.out.println("do something");
}
}
测试类如下:
package go.derek;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
public class Test {
public static void main(String[] args) throws InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
Constructor<?>[] array= EnumSingleton.class.getDeclaredConstructors();
for (Constructor<?> c : array){
c.setAccessible(true);
//此处会抛出异常
c.newInstance();
}
}
}
运行结果如下:
Exception in thread "main" java.lang.IllegalArgumentException: Cannot reflectively create enum objects
at java.lang.reflect.Constructor.newInstance(Constructor.java:521)
at go.derek.Test.main(Test.java:13)
可见,不能通过反射来创建枚举对象,所以这种单例模式可以抵御恶意客户端通过反射的攻击。此外,枚举类的单例模式也不必担心反序列化的时候多次创建实例。