关于单例模式,相信你们都全部了解,比较经典的实现有饿汉式、借助内部类、双重锁检测,这些实现能够保证线程安全,可是在某些特殊状况下并不可以保证仅仅只有一个单例,由于像序列化、反射攻击等每每能够生成新的实例对象,本文将重点分析枚举单例模式如何防止反射攻击。java
枚举单例:安全
public enum Singleton { INSTANCE { @Override protected void read() { System.out.println("read"); } @Override protected void write() { System.out.println("write"); } }; protected abstract void read(); protected abstract void write(); }
以上是一个单例枚举的例子,而咱们要获取该实例只须要Singleton.INSTANCE,而且此种方式能够保证该单例线程安全、防反射攻击、防止序列化生成新的实例。ide
枚举单例关于防反射攻击,固然和枚举的实现有关,枚举也是java类,咱们对Singleton的class进行反编译,能够获得一个新的类(对于枚举的实现不了解的能够补补相关知识):this
反编译后的类:线程
public abstract class Singleton extends Enum { private Singleton(String s, int i) { super(s, i); } protected abstract void read(); protected abstract void write(); public static Singleton[] values() { Singleton asingleton[]; int i; Singleton asingleton1[]; System.arraycopy(asingleton = ENUM$VALUES, 0, asingleton1 = new Singleton[i = asingleton.length], 0, i); return asingleton1; } public static Singleton valueOf(String s) { return (Singleton)Enum.valueOf(singleton/Singleton, s); } Singleton(String s, int i, Singleton singleton) { this(s, i); } public static final Singleton INSTANCE; private static final Singleton ENUM$VALUES[]; static { INSTANCE = new Singleton("INSTANCE", 0) { protected void read() { System.out.println("read"); } protected void write() { System.out.println("write"); } }; ENUM$VALUES = (new Singleton[] { INSTANCE }); } }
看到了这个类的真身事后,相信不少人对于枚举单例防反射的的原理就了解了:code
//readResolve to prevent another instance of Singleton private Object readResolve(){ return INSTANCE; }
若是写的有问题,欢迎指正~对象