單例模式的新實現
jdk1.5 之前 單例模式的兩種方式,兩種方法都是要把構造器保持私有的,并導出公有的靜態成員,以便允許客戶端能夠訪問該類的唯一實例。
第一種方法中,公有的靜態成員是個final域:
//Singleton with public final field
public class Elvis {public static final Elvis INSTANCE = new Elvis();private Elvis (){};public void leaveTheBuilding(){...}
}
私有的構造器僅被調用一次,用來實例公有化的靜態final
域Elvis.INSTANCE
。因為缺少公有的或者受保護的構造器,所以保證了Elvis
的全局唯一性:一旦Elvis
被實例化,只會存在一個實例,不多也不少。
第二種方法中,共有的成員是靜態工廠方法:
//Singleton with static factory
public class Elvis {private static final Elvis INSTANCE = new Elvis();private Elvis (){};public static Elvis getInstance(){return INSTANCE}public void leaveTheBuilding(){...}
}
對于靜態方法Elvis.getInstance
方法的所有調用都會返回同一個人對象引用,所以永遠不會創建其他的Elvis
實例
ps:享有特權的客戶端可以借助AccessibleObject.setAccessible
方法,通過反射機制調用私有的構造器。如果需要防御這種攻擊,可以修改構造器,讓它在被要求創建第二個實例的時候拋出異常。
1.5發行版之后,實現Singleton還有第三種方法。只需要編寫一個包含單個元素的枚舉類型:
//Enum singleton - the preferred approach
public enum Elvis{INSTANCE;public void leaveTheBuilding(){}
}
這種方式在功能上與公有域方法接近,但是它更加簡潔,無償提供了序列化機制,絕對的防止多次實例化,即使是面對復雜的序列化或者反射攻擊的時候。
單元素的枚舉類型已經成為實現Singleton的最佳方法